import gi import os gi.require_version('Gtk', '3.0') from gi.repository import Gtk, GObject from mods.db import Patient, store_patient_index, update_patient_index, search_patients, db from datetime import date import peewee gender_dict = { 'male': 'Мужской', 'female': 'Женский' } doc_type_dict = { 'passport': 'Паспорт', 'birth_cert': 'Св.о рождении', 'foreign_passport': 'Паспорт иностранца' } # Variables resource_dir = os.path.join(os.path.dirname(os.path.abspath(__file__)), 'res') ui_dir = os.path.join(resource_dir, 'ui') icon_dir = os.path.join(resource_dir, 'icons') main_win_file = os.path.join(ui_dir, 'main_win.glade') new_patient_win_file = os.path.join(ui_dir, 'new_patient_win.glade') open_patient_win_file = os.path.join(ui_dir, 'open_patient_win.glade') edit_patient_win_file = os.path.join(ui_dir, 'edit_patient_win.glade') male_patient_row_file = os.path.join(ui_dir, 'male_patient_row.glade') female_patient_row_file = os.path.join(ui_dir, 'female_patient_row.glade') gender_ui_str = {} with open(male_patient_row_file, 'r') as f: gender_ui_str['male'] = f.read() with open(female_patient_row_file, 'r') as f: gender_ui_str['female'] = f.read() ############ builder = Gtk.Builder() class PatientFilter: def __init__(self): self.reset() def filter(self, query): if query != self.fstr: self.fstr = query self.ids = list(map(lambda x: x.id, search_patients(query))) return self.ids def reset(self): self.fstr = '' self.ids = [] patient_filter = PatientFilter() class PatientRow(Gtk.ListBoxRow): @GObject.Property def db_id(self): return self._db_id @db_id.setter def db_id_setter(self, value): self._db_id = value @GObject.Property def fio(self): return self._fio @fio.setter def fio_setter(self, value): self._fio = value @GObject.Property def birth_date(self): return self._birth_date @birth_date.setter def birth_date_setter(self, value): self._birth_date = value class MainWinHandler: def main_win_close(self, *args): Gtk.main_quit() def show_new_patient_win(self, button): new_patient_win = create_new_patient_win() new_patient_win.show_all() def show_open_patient_win(self, button): pl = builder.get_object('patient_list') row = pl.get_selected_row() open_patient_win = create_open_patient_win(row.props.db_id) open_patient_win.show_all() def patient_list_selected(self, *a): button = builder.get_object('patient_open_button') button.set_sensitive(True) def patient_list_unselected(self, *a): button = builder.get_object('patient_open_button') button.set_sensitive(False) def patient_filter_changed(self, filter_widget): pl = builder.get_object('patient_list') pl.unselect_all() self.patient_list_unselected() pl.invalidate_filter() '''pl = builder.get_object('patient_list') for r in pl.get_children(): pl.remove(r) fstr = builder.get_object('patient_filter').get_text().strip() if not fstr: for p in Patient.select(): pl.add(build_patient_row(p)) else: patients = search_patients(fstr) for p in patients: pl.add(build_patient_row(p)) pl.show_all()''' def build_patient_row(patient): b = Gtk.Builder() b.add_from_string(gender_ui_str[patient.gender]) win = b.get_object('win') box = b.get_object('patient_box') icon = b.get_object('icon') icon.set_from_file(os.path.join(icon_dir, f'{patient.gender}.png')) fio = f'{" ".join([patient.last_name, patient.first_name, patient.middle_name])}' b.get_object('fio').set_text(fio) birth_date = patient.birth_date b.get_object('birth_date').set_text(birth_date.strftime('%d.%m.%Y')) win.remove(win.get_children()[0]) row = PatientRow() row.props.db_id = patient.id row.props.fio = fio row.props.birth_date = birth_date row.add(box) return row def patient_sort_func(row1, row2, *a): text1 = row1.props.fio text2 = row2.props.fio return (text1 > text2) - (text1 < text2) def patient_filter_func(row): fstr = builder.get_object('patient_filter').get_text().strip() if not fstr: patient_filter.reset() return True return row.props.db_id in patient_filter.filter(fstr) builder.add_from_file(main_win_file) builder.connect_signals(MainWinHandler()) patient_list = builder.get_object('patient_list') patient_list.set_sort_func(patient_sort_func) patient_list.set_filter_func(patient_filter_func) with db.atomic(): for p in Patient.select(): patient_list.add(build_patient_row(p)) def show_msg(text, sec_text='', level='info'): msg_win_file = os.path.join(ui_dir, f'{level}_win.glade') b = Gtk.Builder() b.add_from_file(msg_win_file) w = b.get_object(level) w.set_property('text',text) w.set_property('secondary_text',sec_text) w.run() w.close() def get_patient_win_values(b): l_name = b.get_object('last_name').get_text().replace(' ', '') f_name = b.get_object('first_name').get_text().replace(' ', '') m_name = b.get_object('middle_name').get_text().replace(' ', '') if '' in (l_name, f_name): show_msg('Не указаны имя или фамилия', 'Данные поля должны быть заполнены', level='warn') return None try: birth_d = int(b.get_object('birth_day').get_text()) if birth_d < 1 or birth_d > 31: return show_msg('Неверный день месяца', 'Укажите число в диапазоне 1-31', level='warn') birth_m = int(b.get_object('birth_month').get_text()) if birth_m < 1 or birth_m > 12: show_msg('Неверный номер месяца', 'Укажите число в диапазоне 1-12', level='warn') return None birth_y = int(b.get_object('birth_year').get_text()) except ValueError: show_msg('Неверно указана дата рождения', 'Все поля даты должны быть заполнены', level='warn') return None gender = b.get_object('gender').get_active_id() if not gender: show_msg('Не выбран пол', level='warn') return None birth_date = date(birth_y, birth_m, birth_d) doc_type = b.get_object('doc_type').get_active_id() doc_serial = b.get_object('doc_serial').get_text() doc_number = b.get_object('doc_number').get_text() policy_number = b.get_object('policy_number').get_text() policy_company = b.get_object('policy_company').get_text() snils_number = b.get_object('snils_number').get_text() notes_buffer = b.get_object('notes_buffer') notes_start = notes_buffer.get_start_iter() notes_end = notes_buffer.get_end_iter() notes = notes_buffer.get_text(notes_start, notes_end, True) return { 'last_name': l_name, 'first_name': f_name, 'middle_name': m_name, 'gender': gender, 'birth_date': birth_date, 'doc_type': doc_type, 'doc_serial': doc_serial, 'doc_number': doc_number, 'policy_number': policy_number, 'policy_company': policy_company, 'snils_number': snils_number, 'notes': notes } def set_patient_values(patient_id, b, edit=False): pat = Patient.select().where(Patient.id == patient_id).get() b.get_object('last_name').set_text(pat.last_name) b.get_object('first_name').set_text(pat.first_name) b.get_object('middle_name').set_text(pat.middle_name) if not edit: b.get_object('gender').set_text(gender_dict[pat.gender]) else: b.get_object('gender').set_active_id(pat.gender) if not edit: b.get_object('birth_date').set_text(pat.birth_date.strftime('%d.%m.%Y')) else: b.get_object('birth_day').set_text(str(pat.birth_date.day)) b.get_object('birth_month').set_text(str(pat.birth_date.month)) b.get_object('birth_year').set_text(str(pat.birth_date.year)) if not edit: b.get_object('doc_type').set_text(doc_type_dict[pat.doc_type] if pat.doc_type else '') elif pat.doc_type: b.get_object('doc_type').set_active_id(pat.doc_type) b.get_object('doc_serial').set_text(pat.doc_serial) b.get_object('doc_number').set_text(pat.doc_number) b.get_object('policy_number').set_text(pat.policy_number) b.get_object('policy_company').set_text(pat.policy_company) b.get_object('snils_number').set_text(pat.snils_number) b.get_object('notes_buffer').set_text(pat.notes) def create_open_patient_win(patient_id): b = Gtk.Builder() class OpenPatientWinHandler: def show_edit_patient_win(self, *a): edit_patient_win = create_edit_patient_win(patient_id) w.destroy() edit_patient_win.show_all() b.add_from_file(open_patient_win_file) b.connect_signals(OpenPatientWinHandler()) w = b.get_object('open_patient_window') # db_id = b.get_object('db_id') # db_id.set_text(str(patient_id)) set_patient_values(patient_id, b) return w def create_edit_patient_win(patient_id): b = Gtk.Builder() class EditPatientWinHandler: def edit_patient_win_close(self, *args): w.destroy() open_patient_win = create_open_patient_win(patient_id) open_patient_win.show_all() def only_digits(self, entry): text = entry.get_text() text = ''.join(filter(lambda x: x.isdigit(), text)) entry.set_text(text) def save_patient(self, *args): values = get_patient_win_values(b) if not values: return with db.atomic(): try: Patient.update(**values).where(Patient.id == patient_id).execute() except peewee.IntegrityError: return show_msg('Данный пациент уже существует', 'Другой пациент с указанными фамилией, именем\nи датой рождения уже есть с базе данных', level='warn') patient = Patient.select().where(Patient.id == patient_id).get() update_patient_index(patient) cur_row = list(filter(lambda x: x.props.db_id == patient_id, patient_list.get_children()))[0] patient_list.remove(cur_row) row = build_patient_row(patient) patient_list.add(row) patient_list.select_row(row) patient_list.show_all() b.get_object('edit_patient_window').close() b.add_from_file(edit_patient_win_file) b.connect_signals(EditPatientWinHandler()) w = b.get_object('edit_patient_window') set_patient_values(patient_id, b, edit=True) return w def create_new_patient_win(): b = Gtk.Builder() class NewPatienWinHandler: def only_digits(self, entry): text = entry.get_text() text = ''.join(filter(lambda x: x.isdigit(), text)) entry.set_text(text) def save_patient(self, *args): values = get_patient_win_values(b) if not values: return with db.atomic(): try: patient = Patient.create(**values) except peewee.IntegrityError: return show_msg('Данный пациент уже существует', 'Пациент с указанными фамилией, именем\nи датой рождения уже есть с базе данных', level='warn') store_patient_index(patient) row = build_patient_row(patient) patient_list.add(row) patient_list.select_row(row) patient_list.show_all() b.get_object('new_patient_window').close() b.add_from_file(new_patient_win_file) b.connect_signals(NewPatienWinHandler()) w = b.get_object('new_patient_window') return w main_win = builder.get_object('main_window') main_win.show_all() Gtk.main()