import gradio as gr import json from datetime import datetime from patient_registration import register_patient from test_selection import select_tests, get_tests_by_category from billing import fetch_billing # Load data def load_data(): try: with open("data.json", "r") as file: data = json.load(file) return data.get("patients", {}), data.get("last_sequence", {"year": None, "month": None, "number": 0}) except FileNotFoundError: return {}, {"year": None, "month": None, "number": 0} # Save data def save_data(patients, last_sequence): with open("data.json", "w") as file: json.dump({"patients": patients, "last_sequence": last_sequence}, file) # Generate patient ID def generate_patient_id(phone, last_sequence): today = datetime.now() current_year = today.year current_month = today.month # Check if year or month has changed if last_sequence["year"] != current_year or last_sequence["month"] != current_month: # Reset sequence for new year/month last_sequence["year"] = current_year last_sequence["month"] = current_month last_sequence["number"] = 1 else: # Increment sequence number last_sequence["number"] += 1 # Format patient ID patient_id = f"{current_year}{current_month:02d}{last_sequence['number']:05d}" return patient_id # Patient Registration Tab def registration_interface(name, father_name, age, phone, address, pincode): patients, last_sequence = load_data() # Generate patient ID patient_id = generate_patient_id(phone, last_sequence) # Add patient details to data patients[patient_id] = { "name": name, "father_name": father_name, "age": age, "phone": phone, "address": address, "pincode": pincode, "tests": [], "total_cost": 0 } # Save data and updated sequence save_data(patients, last_sequence) return f"Patient Registered. Patient ID: {patient_id}" # Tests Selection Tab def test_interface(categories): available_tests = get_tests_by_category(categories) return gr.update(choices=available_tests) # Update dropdown with available tests def confirm_tests_interface(patient_id, selected_tests): patients, last_sequence = load_data() # Load both patients and last_sequence response = select_tests(patient_id, selected_tests, patients) # Update patients directly save_data(patients, last_sequence) # Save both patients and last_sequence return response # Billing Tab def billing_interface(patient_id): patients, _ = load_data() # Load patients and ignore last_sequence if patient_id in patients: billing_info = fetch_billing(patient_id, patients) return billing_info else: return "Invalid Patient ID. Please check the ID." # Gradio Interface with gr.Blocks() as app: gr.Markdown("

\t\t\tSathkrutha LIMS

", elem_id="header") with gr.Tab("Patient Registration"): with gr.Row(): name = gr.Textbox(label="Patient Name", placeholder="Enter patient name", elem_id="name_field", lines=1) father_name = gr.Textbox(label="Father/Husband Name", placeholder="Enter father or husband name", elem_id="father_name_field", lines=1) with gr.Row(): age = gr.Number(label="Age", value=None, minimum=0, maximum=99, elem_id="age_field") phone = gr.Number(label="Phone Number", value=None, elem_id="phone_field") with gr.Row(): address = gr.Textbox(label="Address", placeholder="Enter address", lines=2) pincode = gr.Number(label="Pincode", value=None, elem_id="pincode_field") register_button = gr.Button("Register Patient") registration_output = gr.Textbox(label="Registration Output") register_button.click(registration_interface, [name, father_name, age, phone, address, pincode], registration_output) with gr.Tab("Tests"): patient_id_test = gr.Textbox(label="Patient ID") categories = gr.CheckboxGroup( ["Haematology", "Clinical Pathology", "Biochemistry", "Microbiology", "Specific Diseases", "Serology", "Radiology", "Other Diagnostic Tests"], label="Select Test Categories" ) available_tests = gr.CheckboxGroup(label="Available Tests") confirm_button = gr.Button("Confirm Tests") test_output = gr.Textbox(label="Test Selection Output") categories.change(test_interface, inputs=categories, outputs=available_tests) # Update available tests based on selected categories confirm_button.click(confirm_tests_interface, [patient_id_test, available_tests], test_output) with gr.Tab("Billing"): patient_id_bill = gr.Textbox(label="Patient ID") fetch_button = gr.Button("Fetch Billing") billing_output = gr.Textbox(label="Billing Information") fetch_button.click(billing_interface, [patient_id_bill], billing_output) app.css = """ /* Global styling */ body { font-family: 'Arial', sans-serif; background: url('./background.png') no-repeat center center fixed; background-size: cover; color: #f0f0f0; } /* Header Styling */ #header h1 { color: #ff9800; font-size: 2.5em; font-weight: bold; text-align: center; text-shadow: 2px 2px 4px rgba(0, 0, 0, 0.6); padding-bottom: 20px; } /* Container Styling */ .gradio-container { max-width: 850px; margin: 20px auto; padding: 30px; background: rgba(34, 34, 34, 0.85); border-radius: 12px; box-shadow: 0px 4px 20px rgba(0, 0, 0, 0.5); color: #f0f0f0; } /* Tab Styling with Rounded Rectangle */ .gradio-container .gr-tab { color: #00ff40; padding: 10px 20px; font-weight: bold; cursor: pointer; position: relative; transition: color 0.3s ease, background-color 0.3s ease; border-radius: 20px; /* Rounded corners for tabs */ } .gradio-container .gr-tab:hover { color: #ffffff; background-color: #0000ff; /* Background color for hover effect */ } .gradio-container .gr-tab-active { color: #ffffff; background-color: #ff9800; border-radius: 12px; /* Rounded rectangle for active tab */ } .gradio-container .gr-tab + .gr-tab { margin-left: 10px; /* Adds space between tabs */ } /* Input Field Styling */ #name_field, #father_name_field, #age_field, #phone_field, #pincode_field, textarea { width: 100%; background-color: #2e2e2e; color: #f0f0f0; border: 1px solid #555; border-radius: 8px; padding: 12px; font-size: 1em; transition: border-color 0.3s ease, box-shadow 0.3s ease; margin-top: 10px; } #name_field:focus, #father_name_field:focus, #age_field:focus, #phone_field:focus, #pincode_field:focus, textarea:focus { border-color: #ff9800; box-shadow: 0 0 10px rgba(255, 152, 0, 0.4); outline: none; } /* Button Styling */ .gr-button { background-color: #0000ff; color: #00ff00; border: none; border-radius: 8px; padding: 12px 25px; cursor: pointer; font-size: 1em; font-weight: bold; transition: background-color 0.3s ease, transform 0.2s ease; box-shadow: 0 4px 10px rgba(0, 0, 0, 0.4); margin-top: 15px; } .gr-button:hover { background-color: #00ff00; transform: translateY(-2px); } .gr-button:active { background-color: #00ff00; transform: translateY(0px); } /* Advanced styling for Test Categories and Available Tests checkboxes */ .gr-checkbox-group label { display: inline-flex; align-items: center; padding: 10px 15px; margin: 5px; font-size: 0.9em; font-weight: bold; color: #f0f0f0; background-color: #333; border: 1px solid #555; border-radius: 8px; cursor: pointer; transition: all 0.3s ease; } .gr-checkbox-group label:hover { background-color: #444; border-color: #ff9800; color: #ff9800; } .gr-checkbox-group input[type="checkbox"]:checked + label { background-color: #ff9800; border-color: #ff9800; color: #ffffff; } .gr-checkbox-group input[type="checkbox"] { display: none; } /* Billing Output Styling */ #billing_output { background-color: rgba(176, 201, 30, 0.1); padding: 15px; border-radius: 8px; color: #272dd9; border: 1px solid #b0c91e; font-size: 1em; margin-top: 10px; } /* Responsive Design */ @media (max-width: 768px) { #header h1 { font-size: 2em; } .gradio-container { padding: 20px; } .gr-button { padding: 10px 20px; font-size: 0.9em; } } """ app.launch()