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": 0, "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 if last_sequence["year"] != current_year or last_sequence["month"] != current_month: last_sequence["year"] = current_year last_sequence["month"] = current_month last_sequence["number"] = 1 else: last_sequence["number"] += 1 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, gender, phone, address): patients, last_sequence = load_data() patient_id = generate_patient_id(phone, last_sequence) patients[patient_id] = { "name": name, "father_name": father_name, "age": age, "gender": gender, # Updated field for gender "phone": phone, "address": address, "tests": [], "total_cost": 0 } 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) def confirm_tests_interface(patient_id, selected_tests): patients, last_sequence = load_data() response = select_tests(patient_id, selected_tests, patients) save_data(patients, last_sequence) return response # Billing Tab def billing_interface(patient_id): patients, _ = load_data() if patient_id in patients: billing_info = fetch_billing(patient_id, patients) return billing_info else: return "Invalid Patient ID. Please check the ID." # Custom CSS styling custom_css = """ @import url('https://fonts.googleapis.com/css2?family=Poppins:wght@400;600;700&display=swap'); /* General Page Style */ body { font-family: 'Poppins', Arial, sans-serif; background: linear-gradient(135deg, #4a90e2, #003c7c); color: #ffffff; margin: 0; padding: 0; display: flex; justify-content: center; align-items: center; min-height: 100vh; } /* Heading Colors */ h1 { color: #ffffff; /* White color for h1 */ text-shadow: 2px 2px 10px rgba(0, 0, 0, 0.3); } h2, h3, h4, h5, h6 { color: #f0e68c; /* Light yellow shadow for other headings */ text-shadow: 2px 2px 10px rgba(255, 255, 0, 0.3); } /* Specific style for Age input field */ #age-input { width: 60px; /* Adjust width as needed for a two-digit input */ } /* Header Styling */ #header { color: #7fffd4; text-align: center; font-weight: 700; font-size: 2.5em; margin-bottom: 25px; text-shadow: 2px 2px 10px rgba(0, 0, 0, 0.3); letter-spacing: 1px; transition: color 0.3s ease; } #header:hover { color: #32cd32; transform: scale(1.05); } /* Container Style */ .gradio-container { max-width: 850px; margin: 30px auto; padding: 40px; background: rgba(0, 0, 50, 0.95); border-radius: 15px; box-shadow: 0 10px 25px rgba(0, 0, 0, 0.5); color: #e0f7fa; } /* Form Field Styles */ input, select, textarea { max-width: 700px; border: 1px solid #7fffd4; border-radius: 8px; padding: 10px 12px; background: rgba(245, 245, 245, 0.95); color: #2d2d2d; font-size: 15px; margin-bottom: 20px; transition: all 0.3s ease; box-shadow: 0px 3px 5px rgba(0, 0, 0, 0.15); } /* Radio Button Group Styling */ .gr-radio-group label { display: flex; align-items: center; justify-content: center; padding: 10px 20px; border: 2px solid #7fffd4; border-radius: 8px; transition: all 0.3s ease; cursor: pointer; color: #7fffd4; background: rgba(245, 245, 245, 0.95); box-shadow: 0px 3px 5px rgba(0, 0, 0, 0.15); margin-right: 10px; } .gr-radio-group label:hover { background: #e0f7fa; color: #00bfa5; } /* Radio Button Styling */ .gr-radio-group input[type="radio"] { appearance: none; width: 20px; height: 20px; border: 2px solid #7fffd4; border-radius: 50%; margin-right: 10px; transition: all 0.3s ease; position: relative; } /* Selected Radio Button Styling with Checkmark */ .gr-radio-group input[type="radio"]:checked { background-color: #ff8c00; border-color: #ff8c00; } .gr-radio-group input[type="radio"]:checked::after { content: "โ"; color: #ffffff; font-size: 14px; position: absolute; top: 1px; left: 1px; width: 18px; height: 18px; border-radius: 50%; display: flex; align-items: center; justify-content: center; background-color: #ff8c00; } /* Adjust label for responsive alignment */ .gr-radio-group span { font-size: 1em; font-weight: 600; color: #2d2d2d; } /* Tab Styling */ .nav-tabs { display: flex; justify-content: space-around; background-color: #001848 !important; border-bottom: 1px solid #ffffff !important; padding: 12px !important; gap: 20px !important; } .nav-tabs .nav-item { font-size: 1.2em !important; color: #bfbfbf !important; /* Softer gray for inactive tabs */ padding: 10px 18px !important; cursor: pointer !important; font-weight: 500 !important; border: 2px solid transparent !important; border-radius: 8px !important; transition: color 0.3s, border-color 0.3s !important; } .nav-tabs .nav-item:hover { color: #FFD700 !important; /* Bright yellow on hover */ background-color: rgba(255, 215, 0, 0.1) !important; /* Light yellow background on hover */ } .nav-tabs .nav-item.active { color: #FFD700 !important; /* Bright yellow color for the active tab */ border-bottom: 3px solid #FFD700 !important; /* Bright yellow underline for active tab */ font-weight: 700 !important; background-color: rgba(255, 215, 0, 0.1) !important; /* Light yellow background for active tab */ padding: 8px 16px !important; } /* Output Text Styling */ #registration_output, #test_output, #billing_output { color: #ffffff !important; /* Set text color to white */ background-color: rgba(0, 0, 50, 0.8) !important; padding: 15px; border-radius: 7px; font-weight: bold; text-align: left; } """ # Gradio Interface with gr.Blocks(css=custom_css) as app: gr.Markdown("