@ -0,0 +1,48 @@ | 
				
			|||||
 | 
					.. image:: https://img.shields.io/badge/license-AGPL--3-blue.svg | 
				
			||||
 | 
					   :target: https://www.gnu.org/licenses/agpl-3.0-standalone.html | 
				
			||||
 | 
					   :alt: License: AGPL-3 | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					Dental Clinic Management | 
				
			||||
 | 
					======================== | 
				
			||||
 | 
					Dental Clinic Management is to manage the entire Dental Clinic. | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					Configuration | 
				
			||||
 | 
					============= | 
				
			||||
 | 
					* No additional configuration required | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					License | 
				
			||||
 | 
					------- | 
				
			||||
 | 
					GNU AFFERO GENERAL PUBLIC LICENSE v3.0 (AGPL-3) | 
				
			||||
 | 
					(https://www.gnu.org/licenses/agpl-3.0-standalone.html) | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					Company | 
				
			||||
 | 
					------- | 
				
			||||
 | 
					* `Cybrosys Techno Solutions <https://cybrosys.com/>`__ | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					Credits | 
				
			||||
 | 
					------- | 
				
			||||
 | 
					* Developer: (V17) Kailas Krishna, | 
				
			||||
 | 
					  Contact: odoo@cybrosys.com | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					Contacts | 
				
			||||
 | 
					-------- | 
				
			||||
 | 
					* Mail Contact : odoo@cybrosys.com | 
				
			||||
 | 
					* Website : https://cybrosys.com | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					Bug Tracker | 
				
			||||
 | 
					----------- | 
				
			||||
 | 
					Bugs are tracked on GitHub Issues. In case of trouble, please check there if | 
				
			||||
 | 
					your issue has already been reported. | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					Maintainer | 
				
			||||
 | 
					========== | 
				
			||||
 | 
					.. image:: https://cybrosys.com/images/logo.png | 
				
			||||
 | 
					   :target: https://cybrosys.com | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					This module is maintained by Cybrosys Technologies. | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					For support and more information, please visit `Our Website <https://cybrosys.com/>`__ | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					Further information | 
				
			||||
 | 
					=================== | 
				
			||||
 | 
					HTML Description: `<static/description/index.html>`__ | 
				
			||||
@ -0,0 +1,24 @@ | 
				
			|||||
 | 
					# -*- coding: utf-8 -*- | 
				
			||||
 | 
					################################################################################ | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    Cybrosys Technologies Pvt. Ltd. | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    Copyright (C) 2024-TODAY Cybrosys Technologies(<https://www.cybrosys.com>). | 
				
			||||
 | 
					#    Author: Cybrosys Techno Solutions(<https://www.cybrosys.com>) | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    You can modify it under the terms of the GNU AFFERO | 
				
			||||
 | 
					#    GENERAL PUBLIC LICENSE (AGPL v3), Version 3. | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    This program is distributed in the hope that it will be useful, | 
				
			||||
 | 
					#    but WITHOUT ANY WARRANTY; without even the implied warranty of | 
				
			||||
 | 
					#    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
				
			||||
 | 
					#    GNU AFFERO GENERAL PUBLIC LICENSE (AGPL v3) for more details. | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    You should have received a copy of the GNU AFFERO GENERAL PUBLIC LICENSE | 
				
			||||
 | 
					#    (AGPL v3) along with this program. | 
				
			||||
 | 
					#    If not, see <http://www.gnu.org/licenses/>. | 
				
			||||
 | 
					# | 
				
			||||
 | 
					################################################################################ | 
				
			||||
 | 
					from . import controllers | 
				
			||||
 | 
					from . import models | 
				
			||||
 | 
					from . import wizard | 
				
			||||
@ -0,0 +1,78 @@ | 
				
			|||||
 | 
					# -*- coding: utf-8 -*- | 
				
			||||
 | 
					################################################################################ | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    Cybrosys Technologies Pvt. Ltd. | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    Copyright (C) 2024-TODAY Cybrosys Technologies(<https://www.cybrosys.com>). | 
				
			||||
 | 
					#    Author: Cybrosys Techno Solutions(<https://www.cybrosys.com>) | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    You can modify it under the terms of the GNU AFFERO | 
				
			||||
 | 
					#    GENERAL PUBLIC LICENSE (AGPL v3), Version 3. | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    This program is distributed in the hope that it will be useful, | 
				
			||||
 | 
					#    but WITHOUT ANY WARRANTY; without even the implied warranty of | 
				
			||||
 | 
					#    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
				
			||||
 | 
					#    GNU AFFERO GENERAL PUBLIC LICENSE (AGPL v3) for more details. | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    You should have received a copy of the GNU AFFERO GENERAL PUBLIC LICENSE | 
				
			||||
 | 
					#    (AGPL v3) along with this program. | 
				
			||||
 | 
					#    If not, see <http://www.gnu.org/licenses/>. | 
				
			||||
 | 
					# | 
				
			||||
 | 
					################################################################################ | 
				
			||||
 | 
					{ | 
				
			||||
 | 
					    'name': 'Dental Clinic Management', | 
				
			||||
 | 
					    'version': '17.0.1.0.0', | 
				
			||||
 | 
					    'category': 'Industries', | 
				
			||||
 | 
					    'summary': """Dental Clinic Management is to manage the entire dental clinic.""", | 
				
			||||
 | 
					    'description': """Dental Clinic Management software, dental clinics can  | 
				
			||||
 | 
					    enhance efficiency, improve patient care, optimize resource utilization,  | 
				
			||||
 | 
					    and maintain smooth operations.""", | 
				
			||||
 | 
					    'author': 'Cybrosys Techno Solutions', | 
				
			||||
 | 
					    'company': 'Cybrosys Techno Solutions', | 
				
			||||
 | 
					    'maintainer': 'Cybrosys Techno Solutions', | 
				
			||||
 | 
					    'website': "https://www.cybrosys.com", | 
				
			||||
 | 
					    'depends': ['hr', 'website', 'mail', 'sale_management', 'purchase','stock'], | 
				
			||||
 | 
					    'assets': { | 
				
			||||
 | 
					        'web.assets_frontend': [ | 
				
			||||
 | 
					            "/dental_clinic_management/static/src/js/dental_clinic.js" | 
				
			||||
 | 
					        ] | 
				
			||||
 | 
					    }, | 
				
			||||
 | 
					    'data': [ | 
				
			||||
 | 
					        'security/dental_clinic_management_groups.xml', | 
				
			||||
 | 
					        'security/dental_clinic_management_security.xml', | 
				
			||||
 | 
					        'security/ir.model.access.csv', | 
				
			||||
 | 
					        'data/ir_sequence.xml', | 
				
			||||
 | 
					        'data/dental_department_data.xml', | 
				
			||||
 | 
					        'data/dental_specialist_data.xml', | 
				
			||||
 | 
					        'data/treatment_category_data.xml', | 
				
			||||
 | 
					        'data/dental_treatment_data.xml', | 
				
			||||
 | 
					        'data/insurance_company_data.xml', | 
				
			||||
 | 
					        'data/medicine_frequency_data.xml', | 
				
			||||
 | 
					        'data/dental_time_shift_data.xml', | 
				
			||||
 | 
					        'data/website_menu.xml', | 
				
			||||
 | 
					        'views/dental_time_shift_views.xml', | 
				
			||||
 | 
					        'views/dental_department_views.xml', | 
				
			||||
 | 
					        'views/dental_doctor_views.xml', | 
				
			||||
 | 
					        'views/dental_patients_views.xml', | 
				
			||||
 | 
					        'views/dental_prescription_views.xml', | 
				
			||||
 | 
					        'views/dental_medicine_views.xml', | 
				
			||||
 | 
					        'views/dental_specialist_views.xml', | 
				
			||||
 | 
					        'views/dental_treatment_views.xml', | 
				
			||||
 | 
					        'views/insurance_company_views.xml', | 
				
			||||
 | 
					        'views/medicine_frequency_views.xml', | 
				
			||||
 | 
					        'views/medical_questions_views.xml', | 
				
			||||
 | 
					        'views/treatment_category_views.xml', | 
				
			||||
 | 
					        'views/dental_appointment_views.xml', | 
				
			||||
 | 
					        'views/patient_portal_template.xml', | 
				
			||||
 | 
					        'views/dental_clinic_template.xml', | 
				
			||||
 | 
					        'report/dental_prescription_templates.xml', | 
				
			||||
 | 
					        'report/dental_prescription_report.xml', | 
				
			||||
 | 
					        'wizard/xray_report_views.xml', | 
				
			||||
 | 
					        'views/dental_clinic_management_menu.xml', | 
				
			||||
 | 
					    ], | 
				
			||||
 | 
					    'images': ['static/description/banner.jpg'], | 
				
			||||
 | 
					    'license': 'AGPL-3', | 
				
			||||
 | 
					    'installable': True, | 
				
			||||
 | 
					    'auto_install': False, | 
				
			||||
 | 
					    'application': True, | 
				
			||||
 | 
					} | 
				
			||||
@ -0,0 +1,23 @@ | 
				
			|||||
 | 
					# -*- coding: utf-8 -*- | 
				
			||||
 | 
					################################################################################ | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    Cybrosys Technologies Pvt. Ltd. | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    Copyright (C) 2024-TODAY Cybrosys Technologies(<https://www.cybrosys.com>). | 
				
			||||
 | 
					#    Author: Cybrosys Techno Solutions(<https://www.cybrosys.com>) | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    You can modify it under the terms of the GNU AFFERO | 
				
			||||
 | 
					#    GENERAL PUBLIC LICENSE (AGPL v3), Version 3. | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    This program is distributed in the hope that it will be useful, | 
				
			||||
 | 
					#    but WITHOUT ANY WARRANTY; without even the implied warranty of | 
				
			||||
 | 
					#    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
				
			||||
 | 
					#    GNU AFFERO GENERAL PUBLIC LICENSE (AGPL v3) for more details. | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    You should have received a copy of the GNU AFFERO GENERAL PUBLIC LICENSE | 
				
			||||
 | 
					#    (AGPL v3) along with this program. | 
				
			||||
 | 
					#    If not, see <http://www.gnu.org/licenses/>. | 
				
			||||
 | 
					# | 
				
			||||
 | 
					################################################################################ | 
				
			||||
 | 
					from . import dental_clinic | 
				
			||||
 | 
					from . import patient_portal | 
				
			||||
@ -0,0 +1,100 @@ | 
				
			|||||
 | 
					# -*- coding: utf-8 -*- | 
				
			||||
 | 
					################################################################################ | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    Cybrosys Technologies Pvt. Ltd. | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    Copyright (C) 2024-TODAY Cybrosys Technologies(<https://www.cybrosys.com>). | 
				
			||||
 | 
					#    Author: Cybrosys Techno Solutions(<https://www.cybrosys.com>) | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    You can modify it under the terms of the GNU AFFERO | 
				
			||||
 | 
					#    GENERAL PUBLIC LICENSE (AGPL v3), Version 3. | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    This program is distributed in the hope that it will be useful, | 
				
			||||
 | 
					#    but WITHOUT ANY WARRANTY; without even the implied warranty of | 
				
			||||
 | 
					#    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
				
			||||
 | 
					#    GNU AFFERO GENERAL PUBLIC LICENSE (AGPL v3) for more details. | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    You should have received a copy of the GNU AFFERO GENERAL PUBLIC LICENSE | 
				
			||||
 | 
					#    (AGPL v3) along with this program. | 
				
			||||
 | 
					#    If not, see <http://www.gnu.org/licenses/>. | 
				
			||||
 | 
					# | 
				
			||||
 | 
					################################################################################ | 
				
			||||
 | 
					from odoo.exceptions import UserError | 
				
			||||
 | 
					from odoo import _ | 
				
			||||
 | 
					from odoo.http import Controller, request, route | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					class DentalClinic(Controller): | 
				
			||||
 | 
					    """Controller for a dental clinic website that allows users to view clinic | 
				
			||||
 | 
					    details and schedule appointments online.""" | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					    @route('/dental_doctor', auth='public', website=True) | 
				
			||||
 | 
					    def dental_clinic(self): | 
				
			||||
 | 
					        """Renders the dental clinic page with patient, specialist, and doctor information. | 
				
			||||
 | 
					        This method retrieves the current user's partner ID as the patient ID, | 
				
			||||
 | 
					        fetches all records from the `dental.specialist` model, and all records | 
				
			||||
 | 
					        from the `hr.employee` model to display on the dental clinic webpage.""" | 
				
			||||
 | 
					        patient_id = request.env.user.partner_id | 
				
			||||
 | 
					        specialised_id = request.env['dental.specialist'].sudo().search([]) | 
				
			||||
 | 
					        doctor_id = request.env['hr.employee'].sudo().search([]) | 
				
			||||
 | 
					        return request.render('dental_clinic_management.website_dental_template', | 
				
			||||
 | 
					                              {'patient_id': patient_id, | 
				
			||||
 | 
					                               'specialised_id': specialised_id, | 
				
			||||
 | 
					                               'doctor_id': doctor_id}) | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					    @route('/create/appointment', auth='public', website=True) | 
				
			||||
 | 
					    def create_appointment(self, **kw): | 
				
			||||
 | 
					        """To create a new appointment from website""" | 
				
			||||
 | 
					        if len(kw.get('time_shift')) == 0: | 
				
			||||
 | 
					            raise UserError(_('Doctor Doesnot have the available appointment')) | 
				
			||||
 | 
					        else: | 
				
			||||
 | 
					            patient_appointment = request.env['dental.appointment'].sudo().create({ | 
				
			||||
 | 
					                'patient_id': kw.get('patient'), | 
				
			||||
 | 
					                'patient_phone': kw.get('phone'), | 
				
			||||
 | 
					                'patient_age': kw.get('age'), | 
				
			||||
 | 
					                'specialist_id': kw.get('specialization', False), | 
				
			||||
 | 
					                'doctor_id': kw.get('doctor'), | 
				
			||||
 | 
					                'shift_id': kw.get('time_shift'), | 
				
			||||
 | 
					                'date': kw.get('date'), | 
				
			||||
 | 
					            }) | 
				
			||||
 | 
					            return request.redirect(f'/success_appointment?token={patient_appointment.token_no}') | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					    @route('/success_appointment', auth='public', website=True) | 
				
			||||
 | 
					    def success_appointment(self, **kwargs): | 
				
			||||
 | 
					        """Return when appointment creation is success""" | 
				
			||||
 | 
					        return request.render( | 
				
			||||
 | 
					            'dental_clinic_management.website_rental_success_template', {'token': kwargs}) | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					    @route('/patient_details', type="json", auth='public', website=True) | 
				
			||||
 | 
					    def get_patient_details(self, patient_id): | 
				
			||||
 | 
					        """Retrieve and return details of a specific patient by their ID. | 
				
			||||
 | 
					        This method accesses the `res.partner` model, retrieves a patient | 
				
			||||
 | 
					        record by the given ID, and returns selected fields of the patient | 
				
			||||
 | 
					        such as phone number and age. | 
				
			||||
 | 
					        Args: | 
				
			||||
 | 
					            patient_id (int): The unique identifier of the patient.""" | 
				
			||||
 | 
					        patient = request.env['res.partner'].sudo().browse(int(patient_id)) | 
				
			||||
 | 
					        return patient.read(fields=['phone', 'patient_age']) | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					    @route('/specialised_doctors', type="json", auth='public', website=True) | 
				
			||||
 | 
					    def get_specialised_doctors(self, specialised_id): | 
				
			||||
 | 
					        """To get the list of doctors based on their specialisation""" | 
				
			||||
 | 
					        domain = [] | 
				
			||||
 | 
					        if specialised_id: | 
				
			||||
 | 
					            domain = [('specialised_in_id', '=', int(specialised_id))] | 
				
			||||
 | 
					        doctors = request.env['hr.employee'].sudo().search_read(domain, ["name"]) | 
				
			||||
 | 
					        return doctors | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					    @route('/doctors_shifts', type="json", auth='public', website=True) | 
				
			||||
 | 
					    def get_doctors_shifts(self, doctor_id): | 
				
			||||
 | 
					        """To get the particular doctor time slots""" | 
				
			||||
 | 
					        doctors_shift = request.env['hr.employee'].sudo().browse(int(doctor_id)).time_shift_ids | 
				
			||||
 | 
					        time_shifts = [{"id": rec.id, "name": rec.name} for rec in doctors_shift] | 
				
			||||
 | 
					        return time_shifts | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					    @route('/all_doctors', auth='public', website=True) | 
				
			||||
 | 
					    def get_all_doctors(self): | 
				
			||||
 | 
					        """To list all the doctors""" | 
				
			||||
 | 
					        doctor_id = request.env['hr.employee'].sudo().search([]) | 
				
			||||
 | 
					        return request.render('dental_clinic_management.website_all_doctors', | 
				
			||||
 | 
					                              {'doctor_ids': doctor_id}) | 
				
			||||
@ -0,0 +1,65 @@ | 
				
			|||||
 | 
					# -*- coding: utf-8 -*- | 
				
			||||
 | 
					################################################################################ | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    Cybrosys Technologies Pvt. Ltd. | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    Copyright (C) 2024-TODAY Cybrosys Technologies(<https://www.cybrosys.com>). | 
				
			||||
 | 
					#    Author: Cybrosys Techno Solutions(<https://www.cybrosys.com>) | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    You can modify it under the terms of the GNU AFFERO | 
				
			||||
 | 
					#    GENERAL PUBLIC LICENSE (AGPL v3), Version 3. | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    This program is distributed in the hope that it will be useful, | 
				
			||||
 | 
					#    but WITHOUT ANY WARRANTY; without even the implied warranty of | 
				
			||||
 | 
					#    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
				
			||||
 | 
					#    GNU AFFERO GENERAL PUBLIC LICENSE (AGPL v3) for more details. | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    You should have received a copy of the GNU AFFERO GENERAL PUBLIC LICENSE | 
				
			||||
 | 
					#    (AGPL v3) along with this program. | 
				
			||||
 | 
					#    If not, see <http://www.gnu.org/licenses/>. | 
				
			||||
 | 
					# | 
				
			||||
 | 
					################################################################################ | 
				
			||||
 | 
					from odoo import http | 
				
			||||
 | 
					from odoo.http import request | 
				
			||||
 | 
					from odoo.addons.portal.controllers import portal | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					class PatientPortal(portal.CustomerPortal): | 
				
			||||
 | 
					    """Provide portal access for patients to view their treatment | 
				
			||||
 | 
					    details, prescriptions, and invoices.""" | 
				
			||||
 | 
					    def _prepare_home_portal_values(self, counters): | 
				
			||||
 | 
					        """Extends the base method to include the count of dental prescriptions | 
				
			||||
 | 
					        in the returned dictionary if requested. | 
				
			||||
 | 
					        Args: | 
				
			||||
 | 
					            counters (list): A list of strings indicating which counts to | 
				
			||||
 | 
					            include in the response.""" | 
				
			||||
 | 
					        values = super()._prepare_home_portal_values(counters) | 
				
			||||
 | 
					        if 'prescriptions_count' in counters: | 
				
			||||
 | 
					            prescriptions_count = request.env['dental.prescription'].sudo().search_count([]) | 
				
			||||
 | 
					            values['prescriptions_count'] = prescriptions_count | 
				
			||||
 | 
					        return values | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					    @http.route(['/my/prescriptions'], type='http', auth="user", website=True) | 
				
			||||
 | 
					    def portal_my_prescriptions(self, **kwargs): | 
				
			||||
 | 
					        """Renders the prescriptions page for the logged-in user based on their role. | 
				
			||||
 | 
					        Managers see all prescriptions, doctors see their own, and patients see | 
				
			||||
 | 
					        their own prescriptions.""" | 
				
			||||
 | 
					        if request.env.ref('dental_clinic_management.group_dental_manager') in request.env.user.groups_id: | 
				
			||||
 | 
					            domain = [] | 
				
			||||
 | 
					        elif request.env.ref('dental_clinic_management.group_dental_doctor') in request.env.user.groups_id: | 
				
			||||
 | 
					            domain = [('prescribed_doctor_id', '=', request.env.user.partner_id.employee_ids.id)] | 
				
			||||
 | 
					        else: | 
				
			||||
 | 
					            domain = [('patient_id', '=', request.env.user.partner_id.id)] | 
				
			||||
 | 
					        prescriptions = request.env['dental.prescription'].sudo().search(domain) | 
				
			||||
 | 
					        return request.render("dental_clinic_management.portal_my_prescriptions", | 
				
			||||
 | 
					                              {'prescriptions': prescriptions, 'page_name': 'prescriptions'}) | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					    @http.route(['/view/prescriptions/<int:id>'], | 
				
			||||
 | 
					                type='http', auth="public", website=True) | 
				
			||||
 | 
					    def view_prescriptions(self, id): | 
				
			||||
 | 
					        """View prescriptions based on the provided ID. | 
				
			||||
 | 
					        :param id: The ID of the prescription to view. | 
				
			||||
 | 
					        :return: Rendered template with prescription details.""" | 
				
			||||
 | 
					        prescription = request.env['dental.prescription'].browse(id) | 
				
			||||
 | 
					        return request.render('dental_clinic_management.prescription_portal_template', | 
				
			||||
 | 
					                              {'prescription_details': prescription, 'page_name': 'prescription'}) | 
				
			||||
@ -0,0 +1,15 @@ | 
				
			|||||
 | 
					<?xml version="1.0" encoding="utf-8"?> | 
				
			||||
 | 
					<odoo> | 
				
			||||
 | 
					    <data noupdate="1"> | 
				
			||||
 | 
					        <!-- Dental department data --> | 
				
			||||
 | 
					        <record model="hr.department" id="clinical_department"> | 
				
			||||
 | 
					            <field name="name">Clinical Department</field> | 
				
			||||
 | 
					        </record> | 
				
			||||
 | 
					        <record model="hr.department" id="telehealth_department"> | 
				
			||||
 | 
					            <field name="name">Telehealth Department</field> | 
				
			||||
 | 
					        </record> | 
				
			||||
 | 
					        <record model="hr.department" id="quality_assurance"> | 
				
			||||
 | 
					            <field name="name">Quality Assurance</field> | 
				
			||||
 | 
					        </record> | 
				
			||||
 | 
					    </data> | 
				
			||||
 | 
					</odoo> | 
				
			||||
@ -0,0 +1,18 @@ | 
				
			|||||
 | 
					<?xml version="1.0" encoding="utf-8"?> | 
				
			||||
 | 
					<odoo> | 
				
			||||
 | 
					    <data noupdate="1"> | 
				
			||||
 | 
					        <!-- Data for dental specialist --> | 
				
			||||
 | 
					        <record model="dental.specialist" id="orthodontist_specialist"> | 
				
			||||
 | 
					            <field name="name">Orthodontist</field> | 
				
			||||
 | 
					            <field name="code">ORTHO</field> | 
				
			||||
 | 
					        </record> | 
				
			||||
 | 
					        <record model="dental.specialist" id="periodontist_specialist"> | 
				
			||||
 | 
					            <field name="name">Periodontist</field> | 
				
			||||
 | 
					            <field name="code">PEROD</field> | 
				
			||||
 | 
					        </record> | 
				
			||||
 | 
					        <record model="dental.specialist" id="endodontist_specialist"> | 
				
			||||
 | 
					            <field name="name">Endodontist</field> | 
				
			||||
 | 
					            <field name="code">ENDO</field> | 
				
			||||
 | 
					        </record> | 
				
			||||
 | 
					    </data> | 
				
			||||
 | 
					</odoo> | 
				
			||||
@ -0,0 +1,26 @@ | 
				
			|||||
 | 
					<?xml version="1.0" encoding="utf-8"?> | 
				
			||||
 | 
					<odoo> | 
				
			||||
 | 
					    <data noupdate="1"> | 
				
			||||
 | 
					        <!-- Data for dental time shift --> | 
				
			||||
 | 
					        <record model="dental.time.shift" id="shift_morning"> | 
				
			||||
 | 
					            <field name="shift_type">morning</field> | 
				
			||||
 | 
					            <field name="start_time">06</field> | 
				
			||||
 | 
					            <field name="end_time">09</field> | 
				
			||||
 | 
					        </record> | 
				
			||||
 | 
					        <record model="dental.time.shift" id="shift_evening"> | 
				
			||||
 | 
					            <field name="shift_type">evening</field> | 
				
			||||
 | 
					            <field name="start_time">17</field> | 
				
			||||
 | 
					            <field name="end_time">20</field> | 
				
			||||
 | 
					        </record> | 
				
			||||
 | 
					        <record model="dental.time.shift" id="shift_day"> | 
				
			||||
 | 
					            <field name="shift_type">day</field> | 
				
			||||
 | 
					            <field name="start_time">10</field> | 
				
			||||
 | 
					            <field name="end_time">12</field> | 
				
			||||
 | 
					        </record> | 
				
			||||
 | 
					        <record model="dental.time.shift" id="shift_night"> | 
				
			||||
 | 
					            <field name="shift_type">night</field> | 
				
			||||
 | 
					            <field name="start_time">20</field> | 
				
			||||
 | 
					            <field name="end_time">22</field> | 
				
			||||
 | 
					        </record> | 
				
			||||
 | 
					    </data> | 
				
			||||
 | 
					</odoo> | 
				
			||||
@ -0,0 +1,26 @@ | 
				
			|||||
 | 
					<?xml version="1.0" encoding="utf-8"?> | 
				
			||||
 | 
					<odoo> | 
				
			||||
 | 
					    <data noupdate="1"> | 
				
			||||
 | 
					        <!-- Dental treatment data --> | 
				
			||||
 | 
					        <record model="dental.treatment" id="treatment_teeth_whitening"> | 
				
			||||
 | 
					            <field name="name">Teeth Whitening</field> | 
				
			||||
 | 
					            <field name="treatment_categ_id" ref="cosmetic_dentistry"/> | 
				
			||||
 | 
					        </record> | 
				
			||||
 | 
					        <record model="dental.treatment" id="treatment_gum_contouring"> | 
				
			||||
 | 
					            <field name="name">Gum Contouring</field> | 
				
			||||
 | 
					            <field name="treatment_categ_id" ref="cosmetic_dentistry"/> | 
				
			||||
 | 
					        </record> | 
				
			||||
 | 
					        <record model="dental.treatment" id="treatment_dental_fillings"> | 
				
			||||
 | 
					            <field name="name">Dental Fillings</field> | 
				
			||||
 | 
					            <field name="treatment_categ_id" ref="restorative_dentistry"/> | 
				
			||||
 | 
					        </record> | 
				
			||||
 | 
					        <record model="dental.treatment" id="treatment_dental_cleanings"> | 
				
			||||
 | 
					            <field name="name">Dental Cleanings</field> | 
				
			||||
 | 
					            <field name="treatment_categ_id" ref="preventive_dentistry"/> | 
				
			||||
 | 
					        </record> | 
				
			||||
 | 
					        <record model="dental.treatment" id="treatment_check_ups"> | 
				
			||||
 | 
					            <field name="name">Routine Check-ups and Examinations</field> | 
				
			||||
 | 
					            <field name="treatment_categ_id" ref="preventive_dentistry"/> | 
				
			||||
 | 
					        </record> | 
				
			||||
 | 
					    </data> | 
				
			||||
 | 
					</odoo> | 
				
			||||
@ -0,0 +1,15 @@ | 
				
			|||||
 | 
					<?xml version="1.0" encoding="utf-8"?> | 
				
			||||
 | 
					<odoo> | 
				
			||||
 | 
					    <data noupdate="1"> | 
				
			||||
 | 
					        <!-- Insurance company data --> | 
				
			||||
 | 
					        <record model="insurance.company" id="insurance_company_star_health"> | 
				
			||||
 | 
					            <field name="name">Star Health and Allied Insurance</field> | 
				
			||||
 | 
					        </record> | 
				
			||||
 | 
					        <record model="insurance.company" id="insurance_company_max_bupa"> | 
				
			||||
 | 
					            <field name="name">Max Bupa Health Insurance</field> | 
				
			||||
 | 
					        </record> | 
				
			||||
 | 
					        <record model="insurance.company" id="insurance_company_unitedhealth"> | 
				
			||||
 | 
					            <field name="name">UnitedHealth Group</field> | 
				
			||||
 | 
					        </record> | 
				
			||||
 | 
					    </data> | 
				
			||||
 | 
					</odoo> | 
				
			||||
@ -0,0 +1,19 @@ | 
				
			|||||
 | 
					<?xml version="1.0" encoding="utf-8"?> | 
				
			||||
 | 
					<odoo noupdate="1"> | 
				
			||||
 | 
					    <!--  Sequence for Prescription  --> | 
				
			||||
 | 
					    <record id="seq_dental_prescription" model="ir.sequence"> | 
				
			||||
 | 
					        <field name="name">Dental Prescription</field> | 
				
			||||
 | 
					        <field name="code">dental.prescriptions</field> | 
				
			||||
 | 
					        <field name="prefix">PRN</field> | 
				
			||||
 | 
					        <field name="padding">3</field> | 
				
			||||
 | 
					        <field name="company_id" eval="False"/> | 
				
			||||
 | 
					    </record> | 
				
			||||
 | 
					    <!--  Sequence for Appointment  --> | 
				
			||||
 | 
					    <record id="seq_dental_appointment" model="ir.sequence"> | 
				
			||||
 | 
					        <field name="name">Dental Appointment</field> | 
				
			||||
 | 
					        <field name="code">dental.appointment</field> | 
				
			||||
 | 
					        <field name="prefix">DAP</field> | 
				
			||||
 | 
					        <field name="padding">5</field> | 
				
			||||
 | 
					        <field name="company_id" eval="False"/> | 
				
			||||
 | 
					    </record> | 
				
			||||
 | 
					</odoo> | 
				
			||||
@ -0,0 +1,30 @@ | 
				
			|||||
 | 
					<?xml version="1.0" encoding="utf-8"?> | 
				
			||||
 | 
					<odoo> | 
				
			||||
 | 
					    <data noupdate="1"> | 
				
			||||
 | 
					        <!-- Medical Frequency data --> | 
				
			||||
 | 
					        <record model="medicine.frequency" id="medicine_frequency_bd"> | 
				
			||||
 | 
					            <field name="code">BD</field> | 
				
			||||
 | 
					            <field name="medicament_frequency">Twice a day</field> | 
				
			||||
 | 
					        </record> | 
				
			||||
 | 
					        <record model="medicine.frequency" id="medicine_frequency_od"> | 
				
			||||
 | 
					            <field name="code">OD</field> | 
				
			||||
 | 
					            <field name="medicament_frequency">Once in a day</field> | 
				
			||||
 | 
					        </record> | 
				
			||||
 | 
					        <record model="medicine.frequency" id="medicine_frequency_tds"> | 
				
			||||
 | 
					            <field name="code">TDS</field> | 
				
			||||
 | 
					            <field name="medicament_frequency">Three times a day</field> | 
				
			||||
 | 
					        </record> | 
				
			||||
 | 
					        <record model="medicine.frequency" id="medicine_frequency_qds"> | 
				
			||||
 | 
					            <field name="code">QDS</field> | 
				
			||||
 | 
					            <field name="medicament_frequency">Four times a day</field> | 
				
			||||
 | 
					        </record> | 
				
			||||
 | 
					        <record model="medicine.frequency" id="medicine_frequency_hs"> | 
				
			||||
 | 
					            <field name="code">HS</field> | 
				
			||||
 | 
					            <field name="medicament_frequency">Bed Time</field> | 
				
			||||
 | 
					        </record> | 
				
			||||
 | 
					        <record model="medicine.frequency" id="medicine_frequency_pc"> | 
				
			||||
 | 
					            <field name="code">PC</field> | 
				
			||||
 | 
					            <field name="medicament_frequency">Before Meals</field> | 
				
			||||
 | 
					        </record> | 
				
			||||
 | 
					    </data> | 
				
			||||
 | 
					</odoo> | 
				
			||||
@ -0,0 +1,15 @@ | 
				
			|||||
 | 
					<?xml version="1.0" encoding="utf-8"?> | 
				
			||||
 | 
					<odoo> | 
				
			||||
 | 
					    <data noupdate="1"> | 
				
			||||
 | 
					        <!-- Data for treatment category --> | 
				
			||||
 | 
					        <record model="treatment.category" id="preventive_dentistry"> | 
				
			||||
 | 
					            <field name="name">Preventive Dentistry</field> | 
				
			||||
 | 
					        </record> | 
				
			||||
 | 
					        <record model="treatment.category" id="restorative_dentistry"> | 
				
			||||
 | 
					            <field name="name">Restorative Dentistry</field> | 
				
			||||
 | 
					        </record> | 
				
			||||
 | 
					        <record model="treatment.category" id="cosmetic_dentistry"> | 
				
			||||
 | 
					            <field name="name">Cosmetic Dentistry</field> | 
				
			||||
 | 
					        </record> | 
				
			||||
 | 
					    </data> | 
				
			||||
 | 
					</odoo> | 
				
			||||
@ -0,0 +1,17 @@ | 
				
			|||||
 | 
					<?xml version="1.0" encoding="utf-8"?> | 
				
			||||
 | 
					<odoo> | 
				
			||||
 | 
					    <!-- Appointment Menu in website --> | 
				
			||||
 | 
					    <record id="appointment_menu" model="website.menu"> | 
				
			||||
 | 
					        <field name="name">Appointment</field> | 
				
			||||
 | 
					        <field name="url">/dental_doctor</field> | 
				
			||||
 | 
					        <field name="parent_id" ref="website.main_menu"/> | 
				
			||||
 | 
					        <field name="sequence" type="int">11</field> | 
				
			||||
 | 
					    </record> | 
				
			||||
 | 
					    <!-- Doctors Menu in website --> | 
				
			||||
 | 
					    <record id="doctors_menu" model="website.menu"> | 
				
			||||
 | 
					        <field name="name">Doctors</field> | 
				
			||||
 | 
					        <field name="url">/all_doctors</field> | 
				
			||||
 | 
					        <field name="parent_id" ref="website.main_menu"/> | 
				
			||||
 | 
					        <field name="sequence" type="int">12</field> | 
				
			||||
 | 
					    </record> | 
				
			||||
 | 
					</odoo> | 
				
			||||
@ -0,0 +1,6 @@ | 
				
			|||||
 | 
					## Module <dental_clinic_management> | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					#### 21.03.2024 | 
				
			||||
 | 
					#### Version 17.0.1.0.0 | 
				
			||||
 | 
					#### ADD | 
				
			||||
 | 
					Initial commit for Dental Clinic Management | 
				
			||||
@ -0,0 +1,34 @@ | 
				
			|||||
 | 
					# -*- coding: utf-8 -*- | 
				
			||||
 | 
					################################################################################ | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    Cybrosys Technologies Pvt. Ltd. | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    Copyright (C) 2024-TODAY Cybrosys Technologies(<https://www.cybrosys.com>). | 
				
			||||
 | 
					#    Author: Cybrosys Techno Solutions(<https://www.cybrosys.com>) | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    You can modify it under the terms of the GNU AFFERO | 
				
			||||
 | 
					#    GENERAL PUBLIC LICENSE (AGPL v3), Version 3. | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    This program is distributed in the hope that it will be useful, | 
				
			||||
 | 
					#    but WITHOUT ANY WARRANTY; without even the implied warranty of | 
				
			||||
 | 
					#    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
				
			||||
 | 
					#    GNU AFFERO GENERAL PUBLIC LICENSE (AGPL v3) for more details. | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    You should have received a copy of the GNU AFFERO GENERAL PUBLIC LICENSE | 
				
			||||
 | 
					#    (AGPL v3) along with this program. | 
				
			||||
 | 
					#    If not, see <http://www.gnu.org/licenses/>. | 
				
			||||
 | 
					# | 
				
			||||
 | 
					################################################################################ | 
				
			||||
 | 
					from . import dental_appointment | 
				
			||||
 | 
					from . import dental_doctor | 
				
			||||
 | 
					from . import dental_medicine | 
				
			||||
 | 
					from . import dental_patients | 
				
			||||
 | 
					from . import dental_prescription | 
				
			||||
 | 
					from . import dental_specialist | 
				
			||||
 | 
					from . import dental_time_shift | 
				
			||||
 | 
					from . import dental_treatment | 
				
			||||
 | 
					from . import insurance_company | 
				
			||||
 | 
					from . import medical_questionnaire | 
				
			||||
 | 
					from . import medical_questions | 
				
			||||
 | 
					from . import medicine_frequency | 
				
			||||
 | 
					from . import treatment_category | 
				
			||||
@ -0,0 +1,131 @@ | 
				
			|||||
 | 
					# -*- coding: utf-8 -*- | 
				
			||||
 | 
					################################################################################ | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    Cybrosys Technologies Pvt. Ltd. | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    Copyright (C) 2024-TODAY Cybrosys Technologies(<https://www.cybrosys.com>). | 
				
			||||
 | 
					#    Author: Cybrosys Techno Solutions(<https://www.cybrosys.com>) | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    You can modify it under the terms of the GNU AFFERO | 
				
			||||
 | 
					#    GENERAL PUBLIC LICENSE (AGPL v3), Version 3. | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    This program is distributed in the hope that it will be useful, | 
				
			||||
 | 
					#    but WITHOUT ANY WARRANTY; without even the implied warranty of | 
				
			||||
 | 
					#    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
				
			||||
 | 
					#    GNU AFFERO GENERAL PUBLIC LICENSE (AGPL v3) for more details. | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    You should have received a copy of the GNU AFFERO GENERAL PUBLIC LICENSE | 
				
			||||
 | 
					#    (AGPL v3) along with this program. | 
				
			||||
 | 
					#    If not, see <http://www.gnu.org/licenses/>. | 
				
			||||
 | 
					# | 
				
			||||
 | 
					################################################################################ | 
				
			||||
 | 
					from odoo import api, fields, models, _ | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					class DentalAppointment(models.Model): | 
				
			||||
 | 
					    """Patient dental appointment details""" | 
				
			||||
 | 
					    _name = 'dental.appointment' | 
				
			||||
 | 
					    _description = "Dental Appointment for patients" | 
				
			||||
 | 
					    _inherit = "mail.thread" | 
				
			||||
 | 
					    _rec_name = 'sequence_no' | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					    sequence_no = fields.Char(string='Sequence No', readonly=True, | 
				
			||||
 | 
					                              default=lambda self: _('New'), | 
				
			||||
 | 
					                              copy=False, | 
				
			||||
 | 
					                              help="Sequence number of appointment") | 
				
			||||
 | 
					    token_no = fields.Integer(string='Token No', copy=False, | 
				
			||||
 | 
					                              readonly=True, | 
				
			||||
 | 
					                              help="Token number of the appointments") | 
				
			||||
 | 
					    patient_id = fields.Many2one('res.partner', | 
				
			||||
 | 
					                                 string="Patient Name", | 
				
			||||
 | 
					                                 domain="[('is_patient', '=', True)]", | 
				
			||||
 | 
					                                 copy=False, | 
				
			||||
 | 
					                                 required=True, | 
				
			||||
 | 
					                                 help="Add the patient") | 
				
			||||
 | 
					    patient_phone = fields.Char(related="patient_id.phone", string="Phone", | 
				
			||||
 | 
					                                help="Phone number of the patient") | 
				
			||||
 | 
					    patient_age = fields.Integer(related="patient_id.patient_age", string="Age", | 
				
			||||
 | 
					                                 help="Age of the patient") | 
				
			||||
 | 
					    specialist_id = fields.Many2one('dental.specialist', | 
				
			||||
 | 
					                                    string="Doctors Department", | 
				
			||||
 | 
					                                    help='Choose the doctors department') | 
				
			||||
 | 
					    doctor_ids = fields.Many2many('hr.employee', | 
				
			||||
 | 
					                                  compute='_compute_doctor_ids', | 
				
			||||
 | 
					                                  string="Doctors Data", help="Doctors Data") | 
				
			||||
 | 
					    doctor_id = fields.Many2one('hr.employee', string="Doctor", | 
				
			||||
 | 
					                                required=True, | 
				
			||||
 | 
					                                domain="[('id', 'in', doctor_ids)]", | 
				
			||||
 | 
					                                help="Name the of the doctor") | 
				
			||||
 | 
					    time_shift_ids = fields.Many2many('dental.time.shift', | 
				
			||||
 | 
					                                      string="Time Shift", | 
				
			||||
 | 
					                                      help="Choose the time shift", | 
				
			||||
 | 
					                                      compute='_compute_time_shifts') | 
				
			||||
 | 
					    shift_id = fields.Many2one('dental.time.shift', | 
				
			||||
 | 
					                               string="Booking Time", | 
				
			||||
 | 
					                               domain="[('id','in',time_shift_ids)]", | 
				
			||||
 | 
					                               help="Choose the time shift") | 
				
			||||
 | 
					    date = fields.Date(string="Date", required=True, | 
				
			||||
 | 
					                       default=fields.date.today(), | 
				
			||||
 | 
					                       help="Date when to take appointment for doctor") | 
				
			||||
 | 
					    reason = fields.Text(string="Please describe the reason", | 
				
			||||
 | 
					                         help="Just explain about the reason to take doctor appointment") | 
				
			||||
 | 
					    state = fields.Selection([('draft', 'Draft'), | 
				
			||||
 | 
					                              ('new', 'New Appointment'), | 
				
			||||
 | 
					                              ('done', 'Prescribed'), | 
				
			||||
 | 
					                              ('cancel', 'Cancel')], | 
				
			||||
 | 
					                             default="draft", | 
				
			||||
 | 
					                             string="State", help="state of the appointment") | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					    @api.model | 
				
			||||
 | 
					    def create(self, vals): | 
				
			||||
 | 
					        """Function declared for creating sequence Number for Appointments""" | 
				
			||||
 | 
					        if vals.get('sequence_no', _('New')) == _('New'): | 
				
			||||
 | 
					            vals['sequence_no'] = self.env['ir.sequence'].next_by_code( | 
				
			||||
 | 
					                'dental.appointment') or _('New') | 
				
			||||
 | 
					        last_token = self.search( | 
				
			||||
 | 
					            [('doctor_id', '=', int(vals['doctor_id'])), | 
				
			||||
 | 
					             ('date', '=', vals['date']), | 
				
			||||
 | 
					             ('shift_id', '=', int(vals['shift_id']))], | 
				
			||||
 | 
					            order='id desc', limit=1) | 
				
			||||
 | 
					        vals['token_no'] = last_token.token_no + 1 if last_token else 1 | 
				
			||||
 | 
					        res = super(DentalAppointment, self).create(vals) | 
				
			||||
 | 
					        res.state = 'new' | 
				
			||||
 | 
					        return res | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					    def action_create_appointment(self): | 
				
			||||
 | 
					        """Change the state of the appointment while click create button""" | 
				
			||||
 | 
					        self.state = 'new' | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					    @api.depends('doctor_id') | 
				
			||||
 | 
					    def _compute_time_shifts(self): | 
				
			||||
 | 
					        """To get the doctors time shift""" | 
				
			||||
 | 
					        for record in self: | 
				
			||||
 | 
					            record.time_shift_ids = self.env['dental.time.shift'].search( | 
				
			||||
 | 
					                [('id', 'in', record.doctor_id.time_shift_ids.ids)]).ids | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					    @api.depends('specialist_id') | 
				
			||||
 | 
					    def _compute_doctor_ids(self): | 
				
			||||
 | 
					        """Searching for doctors based on there specialization""" | 
				
			||||
 | 
					        for record in self: | 
				
			||||
 | 
					            if record.specialist_id: | 
				
			||||
 | 
					                record.doctor_ids = self.env['hr.employee'].search( | 
				
			||||
 | 
					                    [('specialised_in_id', '=', record.specialist_id.id)]).ids | 
				
			||||
 | 
					            else: | 
				
			||||
 | 
					                record.doctor_ids = self.env['hr.employee'].search([]).ids | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					    def action_cancel(self): | 
				
			||||
 | 
					        """Change the state of the appointment while click cancel button""" | 
				
			||||
 | 
					        self.state = 'cancel' | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					    def action_prescription(self): | 
				
			||||
 | 
					        """Created the action for view the prescriptions | 
				
			||||
 | 
					        of 'done' state appointments""" | 
				
			||||
 | 
					        return { | 
				
			||||
 | 
					            'type': 'ir.actions.act_window', | 
				
			||||
 | 
					            'target': 'inline', | 
				
			||||
 | 
					            'name': 'Prescription', | 
				
			||||
 | 
					            'view_mode': 'form', | 
				
			||||
 | 
					            'res_model': 'dental.prescription', | 
				
			||||
 | 
					            'res_id': self.env['dental.prescription'].search([ | 
				
			||||
 | 
					                ('appointment_id', '=', self.id)], limit=1).id, | 
				
			||||
 | 
					        } | 
				
			||||
@ -0,0 +1,65 @@ | 
				
			|||||
 | 
					# -*- coding: utf-8 -*- | 
				
			||||
 | 
					################################################################################ | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    Cybrosys Technologies Pvt. Ltd. | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    Copyright (C) 2024-TODAY Cybrosys Technologies(<https://www.cybrosys.com>). | 
				
			||||
 | 
					#    Author: Cybrosys Techno Solutions(<https://www.cybrosys.com>) | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    You can modify it under the terms of the GNU AFFERO | 
				
			||||
 | 
					#    GENERAL PUBLIC LICENSE (AGPL v3), Version 3. | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    This program is distributed in the hope that it will be useful, | 
				
			||||
 | 
					#    but WITHOUT ANY WARRANTY; without even the implied warranty of | 
				
			||||
 | 
					#    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
				
			||||
 | 
					#    GNU AFFERO GENERAL PUBLIC LICENSE (AGPL v3) for more details. | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    You should have received a copy of the GNU AFFERO GENERAL PUBLIC LICENSE | 
				
			||||
 | 
					#    (AGPL v3) along with this program. | 
				
			||||
 | 
					#    If not, see <http://www.gnu.org/licenses/>. | 
				
			||||
 | 
					# | 
				
			||||
 | 
					################################################################################ | 
				
			||||
 | 
					from odoo import api, fields, models | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					class DentalDoctor(models.Model): | 
				
			||||
 | 
					    """To add the doctors of the clinic""" | 
				
			||||
 | 
					    _inherit = 'hr.employee' | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					    job_position = fields.Char(string="Designation", | 
				
			||||
 | 
					                               help="To add the job position of the doctor") | 
				
			||||
 | 
					    specialised_in_id = fields.Many2one('dental.specialist', | 
				
			||||
 | 
					                                        string='Specialised In', | 
				
			||||
 | 
					                                        help="Add the doctor specialised") | 
				
			||||
 | 
					    dob = fields.Date(string="Date of Birth", | 
				
			||||
 | 
					                      required=True, | 
				
			||||
 | 
					                      help="DOB of the patient") | 
				
			||||
 | 
					    doctor_age = fields.Integer(compute='_compute_doctor_age', | 
				
			||||
 | 
					                                store=True, | 
				
			||||
 | 
					                                string="Age", | 
				
			||||
 | 
					                                help="Age of the patient") | 
				
			||||
 | 
					    sex = fields.Selection([('male', 'Male'), | 
				
			||||
 | 
					                            ('female', 'Female')], | 
				
			||||
 | 
					                           string="Sex", | 
				
			||||
 | 
					                           help="Sex of the patient") | 
				
			||||
 | 
					    time_shift_ids = fields.Many2many('dental.time.shift', | 
				
			||||
 | 
					                                      string="Time Shift", | 
				
			||||
 | 
					                                      help="Time shift of the doctor") | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					    def unlink(self): | 
				
			||||
 | 
					        """Delete the corresponding user from res.users while | 
				
			||||
 | 
					        deleting the doctor""" | 
				
			||||
 | 
					        for record in self: | 
				
			||||
 | 
					            self.env['res.users'].search([('id', '=', record.user_id.id)]).unlink() | 
				
			||||
 | 
					        res = super(DentalDoctor, self).unlink() | 
				
			||||
 | 
					        return res | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					    @api.depends('dob') | 
				
			||||
 | 
					    def _compute_doctor_age(self): | 
				
			||||
 | 
					        """To calculate the age of the doctor from the DOB""" | 
				
			||||
 | 
					        for record in self: | 
				
			||||
 | 
					            record.doctor_age = (fields.date.today().year - record.dob.year - | 
				
			||||
 | 
					                                  ((fields.date.today().month, | 
				
			||||
 | 
					                                    fields.date.today().day) < | 
				
			||||
 | 
					                                   (record.dob.month, | 
				
			||||
 | 
					                                    record.dob.day))) if record.dob else False | 
				
			||||
@ -0,0 +1,36 @@ | 
				
			|||||
 | 
					# -*- coding: utf-8 -*- | 
				
			||||
 | 
					################################################################################ | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    Cybrosys Technologies Pvt. Ltd. | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    Copyright (C) 2024-TODAY Cybrosys Technologies(<https://www.cybrosys.com>). | 
				
			||||
 | 
					#    Author: Cybrosys Techno Solutions(<https://www.cybrosys.com>) | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    You can modify it under the terms of the GNU AFFERO | 
				
			||||
 | 
					#    GENERAL PUBLIC LICENSE (AGPL v3), Version 3. | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    This program is distributed in the hope that it will be useful, | 
				
			||||
 | 
					#    but WITHOUT ANY WARRANTY; without even the implied warranty of | 
				
			||||
 | 
					#    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
				
			||||
 | 
					#    GNU AFFERO GENERAL PUBLIC LICENSE (AGPL v3) for more details. | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    You should have received a copy of the GNU AFFERO GENERAL PUBLIC LICENSE | 
				
			||||
 | 
					#    (AGPL v3) along with this program. | 
				
			||||
 | 
					#    If not, see <http://www.gnu.org/licenses/>. | 
				
			||||
 | 
					# | 
				
			||||
 | 
					################################################################################ | 
				
			||||
 | 
					from odoo import fields, models | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					class DentalMedicine(models.Model): | 
				
			||||
 | 
					    """For creating the medicines used in the dental clinic""" | 
				
			||||
 | 
					    _inherit = 'product.template' | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					    is_medicine = fields.Boolean('Is Medicine', | 
				
			||||
 | 
					                                 help="If the product is a Medicine") | 
				
			||||
 | 
					    generic_name = fields.Char(string="Generic Name", | 
				
			||||
 | 
					                               required=True, | 
				
			||||
 | 
					                               help="Generic name of the medicament") | 
				
			||||
 | 
					    dosage_strength = fields.Integer(string="Dosage Strength", | 
				
			||||
 | 
					                                     required=True, | 
				
			||||
 | 
					                                     help="Dosage strength of medicament") | 
				
			||||
@ -0,0 +1,121 @@ | 
				
			|||||
 | 
					# -*- coding: utf-8 -*- | 
				
			||||
 | 
					################################################################################ | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    Cybrosys Technologies Pvt. Ltd. | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    Copyright (C) 2024-TODAY Cybrosys Technologies(<https://www.cybrosys.com>). | 
				
			||||
 | 
					#    Author: Cybrosys Techno Solutions(<https://www.cybrosys.com>) | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    You can modify it under the terms of the GNU AFFERO | 
				
			||||
 | 
					#    GENERAL PUBLIC LICENSE (AGPL v3), Version 3. | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    This program is distributed in the hope that it will be useful, | 
				
			||||
 | 
					#    but WITHOUT ANY WARRANTY; without even the implied warranty of | 
				
			||||
 | 
					#    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
				
			||||
 | 
					#    GNU AFFERO GENERAL PUBLIC LICENSE (AGPL v3) for more details. | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    You should have received a copy of the GNU AFFERO GENERAL PUBLIC LICENSE | 
				
			||||
 | 
					#    (AGPL v3) along with this program. | 
				
			||||
 | 
					#    If not, see <http://www.gnu.org/licenses/>. | 
				
			||||
 | 
					# | 
				
			||||
 | 
					################################################################################ | 
				
			||||
 | 
					from odoo import api, fields, models, _ | 
				
			||||
 | 
					from odoo.exceptions import UserError | 
				
			||||
 | 
					from odoo.tools import email_normalize | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					class DentalPatients(models.Model): | 
				
			||||
 | 
					    """To create Patients in the clinic, use res.partner model and customize it""" | 
				
			||||
 | 
					    _inherit = 'res.partner' | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					    company_type = fields.Selection(selection_add=[('person', 'Patient'), | 
				
			||||
 | 
					                                                   ('company', 'Medicine Distibutor')], | 
				
			||||
 | 
					                                    help="Patient type") | 
				
			||||
 | 
					    dob = fields.Date(string="Date of Birth", | 
				
			||||
 | 
					                      help="DOB of the patient") | 
				
			||||
 | 
					    patient_age = fields.Integer(compute='_compute_patient_age', | 
				
			||||
 | 
					                                 store=True, | 
				
			||||
 | 
					                                 string="Age", | 
				
			||||
 | 
					                                 help="Age of the patient") | 
				
			||||
 | 
					    sex = fields.Selection([('male', 'Male'), ('female', 'Female')], | 
				
			||||
 | 
					                           string="Sex", | 
				
			||||
 | 
					                           help="Sex of the patient") | 
				
			||||
 | 
					    insurance_company_id = fields.Many2one('insurance.company', | 
				
			||||
 | 
					                                           string="Insurance Company", | 
				
			||||
 | 
					                                           help="Mention the insurance company") | 
				
			||||
 | 
					    start_date = fields.Date(string="Member Since", | 
				
			||||
 | 
					                             help="Patient insurance start date") | 
				
			||||
 | 
					    expiration_date = fields.Date(string="Expiration Date", | 
				
			||||
 | 
					                                  help="Patient insurance expiration date") | 
				
			||||
 | 
					    insureds_name = fields.Char(string="Insured's Name", | 
				
			||||
 | 
					                                help="Name of the insured's") | 
				
			||||
 | 
					    identification_number = fields.Char(string="Identification Number", | 
				
			||||
 | 
					                                        help="Identification Number of insured's") | 
				
			||||
 | 
					    is_patient = fields.Boolean(string="Is Patient", | 
				
			||||
 | 
					                                help="To set it's a patient") | 
				
			||||
 | 
					    medical_questionnaire_ids = fields.One2many('medical.questionnaire', | 
				
			||||
 | 
					                                                 'patient_id', | 
				
			||||
 | 
					                                                readonly=False, | 
				
			||||
 | 
					                                                help="connect model medical questionnaire in patients") | 
				
			||||
 | 
					    report_ids = fields.One2many('xray.report','patient_id', | 
				
			||||
 | 
					                                 string='X-Ray', | 
				
			||||
 | 
					                                 help="To add the xray reports of the patient") | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					    @api.model | 
				
			||||
 | 
					    def create(self, vals): | 
				
			||||
 | 
					        """Overrides the create method to handle additional logic for DentalPatients. | 
				
			||||
 | 
					        When a new DentalPatient is created, It then proceeds to create a portal | 
				
			||||
 | 
					        wizard for the patient to grant them access to the portal. | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					        If the `company_type` is not `person`, it assumes the record is for a | 
				
			||||
 | 
					        Medicine Distributor or another entity. In this case, it creates a user | 
				
			||||
 | 
					        from a template with predefined groups and permissions, and normalizes | 
				
			||||
 | 
					        the email address for consistency.""" | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					        if 'company_type' in vals and vals['company_type'] == 'person': | 
				
			||||
 | 
					            vals['is_patient'] = True | 
				
			||||
 | 
					        res = super(DentalPatients, self).create(vals) | 
				
			||||
 | 
					        if 'company_type' in vals and vals['company_type'] == 'person': | 
				
			||||
 | 
					            wizard = self.env['portal.wizard'].create({ | 
				
			||||
 | 
					                'partner_ids': [fields.Command.link(res.id)] | 
				
			||||
 | 
					            }) | 
				
			||||
 | 
					            portal_wizard = self.env['portal.wizard.user'].sudo().create({ | 
				
			||||
 | 
					                'partner_id': res.id, | 
				
			||||
 | 
					                'email': res.email, | 
				
			||||
 | 
					                'wizard_id': wizard.id, | 
				
			||||
 | 
					            }) | 
				
			||||
 | 
					            portal_wizard.action_grant_access() | 
				
			||||
 | 
					        else: | 
				
			||||
 | 
					            try: | 
				
			||||
 | 
					                user = self.env['res.users'].with_context(no_reset_password=True)._create_user_from_template({ | 
				
			||||
 | 
					                    'email': email_normalize(res.email), | 
				
			||||
 | 
					                    'login': email_normalize(res.email), | 
				
			||||
 | 
					                    'partner_id': res.id, | 
				
			||||
 | 
					                    'groups_id': [ | 
				
			||||
 | 
					                        self.env.ref("base.group_user").id, | 
				
			||||
 | 
					                        self.env.ref('dental_clinic_management.group_dental_doctor').id, | 
				
			||||
 | 
					                        self.env.ref('sales_team.group_sale_salesman').id, | 
				
			||||
 | 
					                        self.env.ref('hr.group_hr_user').id, | 
				
			||||
 | 
					                        self.env.ref('account.group_account_invoice').id, | 
				
			||||
 | 
					                        self.env.ref('stock.group_stock_user').id, | 
				
			||||
 | 
					                        self.env.ref('purchase.group_purchase_user').id | 
				
			||||
 | 
					                    ], | 
				
			||||
 | 
					                    'company_id': self.env.company.id, | 
				
			||||
 | 
					                    'company_ids': [(6, 0, self.env.company.ids)], | 
				
			||||
 | 
					                }) | 
				
			||||
 | 
					                self.env['hr.employee'].search([('work_email', '=', res.email)]).user_id = user.id | 
				
			||||
 | 
					            except: | 
				
			||||
 | 
					                raise UserError(_("Email already used for another dentist")) | 
				
			||||
 | 
					        return res | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					    @api.depends('dob') | 
				
			||||
 | 
					    def _compute_patient_age(self): | 
				
			||||
 | 
					        """Computes the age of the patient based on their date of birth (dob) | 
				
			||||
 | 
					        and updates the `patient_age` field. The age is calculated by subtracting | 
				
			||||
 | 
					        the year of the patient's dob from the current year. If the current | 
				
			||||
 | 
					        date is before the patient's birthday in the current year, one year is | 
				
			||||
 | 
					        subtracted from the age.""" | 
				
			||||
 | 
					        for record in self: | 
				
			||||
 | 
					            record.patient_age = (fields.date.today().year - record.dob.year - | 
				
			||||
 | 
					                                  ((fields.date.today().month,fields.date.today().day) < | 
				
			||||
 | 
					                                   (record.dob.month,record.dob.day))) if record.dob else False | 
				
			||||
@ -0,0 +1,221 @@ | 
				
			|||||
 | 
					# -*- coding: utf-8 -*- | 
				
			||||
 | 
					################################################################################ | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    Cybrosys Technologies Pvt. Ltd. | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    Copyright (C) 2024-TODAY Cybrosys Technologies(<https://www.cybrosys.com>). | 
				
			||||
 | 
					#    Author: Cybrosys Techno Solutions(<https://www.cybrosys.com>) | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    You can modify it under the terms of the GNU AFFERO | 
				
			||||
 | 
					#    GENERAL PUBLIC LICENSE (AGPL v3), Version 3. | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    This program is distributed in the hope that it will be useful, | 
				
			||||
 | 
					#    but WITHOUT ANY WARRANTY; without even the implied warranty of | 
				
			||||
 | 
					#    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
				
			||||
 | 
					#    GNU AFFERO GENERAL PUBLIC LICENSE (AGPL v3) for more details. | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    You should have received a copy of the GNU AFFERO GENERAL PUBLIC LICENSE | 
				
			||||
 | 
					#    (AGPL v3) along with this program. | 
				
			||||
 | 
					#    If not, see <http://www.gnu.org/licenses/>. | 
				
			||||
 | 
					# | 
				
			||||
 | 
					################################################################################ | 
				
			||||
 | 
					from odoo import api, fields, models, _ | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					class DentalPrescription(models.Model): | 
				
			||||
 | 
					    """Prescription of patient from the dental clinic""" | 
				
			||||
 | 
					    _name = 'dental.prescription' | 
				
			||||
 | 
					    _description = "Dental Prescription" | 
				
			||||
 | 
					    _inherit = ['mail.thread'] | 
				
			||||
 | 
					    _rec_name = "sequence_no" | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					    sequence_no = fields.Char(string='Sequence No', required=True, | 
				
			||||
 | 
					                              readonly=True, default=lambda self: _('New'), | 
				
			||||
 | 
					                              help="Sequence number of the dental prescription") | 
				
			||||
 | 
					    appointment_ids = fields.Many2many('dental.appointment', | 
				
			||||
 | 
					                                       string="Appointment", | 
				
			||||
 | 
					                                       compute="_compute_appointment_ids", | 
				
			||||
 | 
					                                       help="All appointments created") | 
				
			||||
 | 
					    appointment_id = fields.Many2one('dental.appointment', | 
				
			||||
 | 
					                                     string="Appointment", | 
				
			||||
 | 
					                                     domain="[('id','in',appointment_ids)]", | 
				
			||||
 | 
					                                     required=True, | 
				
			||||
 | 
					                                     help="All appointments created") | 
				
			||||
 | 
					    patient_id = fields.Many2one(related="appointment_id.patient_id", | 
				
			||||
 | 
					                                 string="Patient", | 
				
			||||
 | 
					                                 required=True, | 
				
			||||
 | 
					                                 help="name of the patient") | 
				
			||||
 | 
					    token_no = fields.Integer(related="appointment_id.token_no", | 
				
			||||
 | 
					                              string="Token Number", | 
				
			||||
 | 
					                              help="Token number of the patient") | 
				
			||||
 | 
					    treatment_id = fields.Many2one('dental.treatment', | 
				
			||||
 | 
					                                   string="Treatment", | 
				
			||||
 | 
					                                   help="Name of the treatment done for patient") | 
				
			||||
 | 
					    cost = fields.Float(related="treatment_id.cost", | 
				
			||||
 | 
					                        string="Treatment Cost", | 
				
			||||
 | 
					                        help="Cost of treatment") | 
				
			||||
 | 
					    currency_id = fields.Many2one('res.currency', 'Currency', | 
				
			||||
 | 
					                                  default=lambda self: self.env.user.company_id.currency_id, | 
				
			||||
 | 
					                                  required=True, | 
				
			||||
 | 
					                                  help="To add the currency type in cost") | 
				
			||||
 | 
					    prescribed_doctor_id = fields.Many2one(related="appointment_id.doctor_id", | 
				
			||||
 | 
					                                           string='Prescribed Doctor', | 
				
			||||
 | 
					                                           required=True, | 
				
			||||
 | 
					                                           help="Doctor who is prescribed") | 
				
			||||
 | 
					    prescription_date = fields.Date(related="appointment_id.date", | 
				
			||||
 | 
					                                    string='Prescription Date', | 
				
			||||
 | 
					                                    required=True, | 
				
			||||
 | 
					                                    help="Date of the prescription") | 
				
			||||
 | 
					    state = fields.Selection([('new', 'New'), | 
				
			||||
 | 
					                              ('done', 'Prescribed'), | 
				
			||||
 | 
					                              ('invoiced', 'Invoiced')], | 
				
			||||
 | 
					                             default="new", | 
				
			||||
 | 
					                             string="state", | 
				
			||||
 | 
					                             help="state of the appointment") | 
				
			||||
 | 
					    medicine_ids = fields.One2many('dental.prescription_lines', | 
				
			||||
 | 
					                                   'prescription_id', | 
				
			||||
 | 
					                                   string="Medicine", | 
				
			||||
 | 
					                                   help="medicines") | 
				
			||||
 | 
					    invoice_data_id = fields.Many2one(comodel_name="account.move", string="Invoice Data", | 
				
			||||
 | 
					                                      help="Invoice Data") | 
				
			||||
 | 
					    grand_total = fields.Float(compute="_compute_grand_total", | 
				
			||||
 | 
					                               string="Grand Total", | 
				
			||||
 | 
					                               help="Get the grand total amount") | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					    @api.model | 
				
			||||
 | 
					    def create(self, vals): | 
				
			||||
 | 
					        """Function declared for creating sequence Number for patients""" | 
				
			||||
 | 
					        if vals.get('sequence_no', _('New')) == _('New'): | 
				
			||||
 | 
					            vals['sequence_no'] = self.env['ir.sequence'].next_by_code( | 
				
			||||
 | 
					                'dental.prescriptions') or _('New') | 
				
			||||
 | 
					        res = super(DentalPrescription, self).create(vals) | 
				
			||||
 | 
					        return res | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					    @api.depends('appointment_id') | 
				
			||||
 | 
					    def _compute_appointment_ids(self): | 
				
			||||
 | 
					        """Computes and assigns the `appointment_ids` field for each record. | 
				
			||||
 | 
					        This method searches for all `dental.appointment` records that have | 
				
			||||
 | 
					        a state of `new` and a date equal to today's date. It then updates | 
				
			||||
 | 
					        the `appointment_ids` field of each `DentalPrescription` record | 
				
			||||
 | 
					        with the IDs of these found appointments.""" | 
				
			||||
 | 
					        for rec in self: | 
				
			||||
 | 
					            rec.appointment_ids = self.env['dental.appointment'].search( | 
				
			||||
 | 
					                [('state', '=', 'new'), ('date', '=', fields.Date.today())]).ids | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					    def action_prescribed(self): | 
				
			||||
 | 
					        """Marks the prescription and its associated appointment as `done`. | 
				
			||||
 | 
					        This method updates the state of both the DentalPrescription instance | 
				
			||||
 | 
					        and its linked dental.appointment instance to `done`, indicating that | 
				
			||||
 | 
					        the prescription has been finalized and the appointment has been completed. | 
				
			||||
 | 
					        """ | 
				
			||||
 | 
					        self.state = 'done' | 
				
			||||
 | 
					        self.appointment_id.state = 'done' | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					    def create_invoice(self): | 
				
			||||
 | 
					        """Create an invoice based on the patient invoice.""" | 
				
			||||
 | 
					        self.ensure_one() | 
				
			||||
 | 
					        invoice_vals = { | 
				
			||||
 | 
					            'move_type': 'out_invoice', | 
				
			||||
 | 
					            'partner_id': self.patient_id.id, | 
				
			||||
 | 
					            'invoice_line_ids': [ | 
				
			||||
 | 
					                fields.Command.create({ | 
				
			||||
 | 
					                    'name': self.treatment_id.name, | 
				
			||||
 | 
					                    'quantity': 1, | 
				
			||||
 | 
					                    'price_unit': self.cost, | 
				
			||||
 | 
					                }) | 
				
			||||
 | 
					            ] | 
				
			||||
 | 
					        } | 
				
			||||
 | 
					        invoice = self.env['account.move'].create(invoice_vals) | 
				
			||||
 | 
					        for rec in self.medicine_ids: | 
				
			||||
 | 
					            product_id = self.env['product.product'].search([ | 
				
			||||
 | 
					                ('product_tmpl_id', '=', rec.medicament_id.id)]) | 
				
			||||
 | 
					            invoice['invoice_line_ids'] = [(0, 0, { | 
				
			||||
 | 
					                'product_id': product_id.id, | 
				
			||||
 | 
					                'name': rec.display_name, | 
				
			||||
 | 
					                'quantity': rec.quantity, | 
				
			||||
 | 
					                'price_unit': rec.price, | 
				
			||||
 | 
					            })] | 
				
			||||
 | 
					        self.invoice_data_id = invoice.id | 
				
			||||
 | 
					        invoice.action_post() | 
				
			||||
 | 
					        self.state = 'invoiced' | 
				
			||||
 | 
					        return { | 
				
			||||
 | 
					            'name': _('Customer Invoice'), | 
				
			||||
 | 
					            'view_mode': 'form', | 
				
			||||
 | 
					            'view_id': self.env.ref('account.view_move_form').id, | 
				
			||||
 | 
					            'res_model': 'account.move', | 
				
			||||
 | 
					            'context': "{'move_type':'out_invoice'}", | 
				
			||||
 | 
					            'type': 'ir.actions.act_window', | 
				
			||||
 | 
					            'res_id': self.invoice_data_id.id, | 
				
			||||
 | 
					        } | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					    def action_view_invoice(self): | 
				
			||||
 | 
					        """Invoice view""" | 
				
			||||
 | 
					        return { | 
				
			||||
 | 
					            'name': _('Customer Invoice'), | 
				
			||||
 | 
					            'view_mode': 'form', | 
				
			||||
 | 
					            'view_id': self.env.ref('account.view_move_form').id, | 
				
			||||
 | 
					            'res_model': 'account.move', | 
				
			||||
 | 
					            'context': "{'move_type':'out_invoice'}", | 
				
			||||
 | 
					            'type': 'ir.actions.act_window', | 
				
			||||
 | 
					            'res_id': self.invoice_data_id.id, | 
				
			||||
 | 
					        } | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					    def _compute_grand_total(self): | 
				
			||||
 | 
					        """Computes the grand total cost of the dental prescription. | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					        This method initializes the grand total with the cost of the treatment | 
				
			||||
 | 
					        and then iterates over all the prescribed medicines, adding their total | 
				
			||||
 | 
					        cost to the grand total. The grand total is stored in the `grand_total` | 
				
			||||
 | 
					        field of the `DentalPrescription` model.""" | 
				
			||||
 | 
					        self.grand_total = self.cost | 
				
			||||
 | 
					        for rec in self.medicine_ids: | 
				
			||||
 | 
					            self.grand_total += rec.total | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					class DentalPrescriptionLines(models.Model): | 
				
			||||
 | 
					    """Prescription lines of the dental clinic prescription""" | 
				
			||||
 | 
					    _name = 'dental.prescription_lines' | 
				
			||||
 | 
					    _description = "Dental Prescriptions Lines" | 
				
			||||
 | 
					    _rec_name = "medicament_id" | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					    medicament_id = fields.Many2one('product.template', | 
				
			||||
 | 
					                                    domain="[('is_medicine', '=', True)]", | 
				
			||||
 | 
					                                    string="Medicament", | 
				
			||||
 | 
					                                    help="Name of the medicine") | 
				
			||||
 | 
					    generic_name = fields.Char(string="Generic Name", | 
				
			||||
 | 
					                               related="medicament_id.generic_name", | 
				
			||||
 | 
					                               help="Generic name of the medicament") | 
				
			||||
 | 
					    dosage_strength = fields.Integer(string="Dosage Strength", | 
				
			||||
 | 
					                                     related="medicament_id.dosage_strength", | 
				
			||||
 | 
					                                     help="Dosage strength of medicament") | 
				
			||||
 | 
					    medicament_form = fields.Selection([('tablet', 'Tablets'), | 
				
			||||
 | 
					                             ('capsule', 'Capsules'), | 
				
			||||
 | 
					                             ('liquid', 'Liquid'), | 
				
			||||
 | 
					                             ('injection', 'Injections')], | 
				
			||||
 | 
					                            string="Medicament Form", | 
				
			||||
 | 
					                            required=True, | 
				
			||||
 | 
					                            help="Add the form of the medicine") | 
				
			||||
 | 
					    quantity = fields.Integer(string="Quantity", | 
				
			||||
 | 
					                              required=True, | 
				
			||||
 | 
					                              help="Quantity of medicine") | 
				
			||||
 | 
					    frequency_id = fields.Many2one('medicine.frequency', | 
				
			||||
 | 
					                                   string="Frequency", | 
				
			||||
 | 
					                                   required=True, | 
				
			||||
 | 
					                                   help="Frequency of medicine") | 
				
			||||
 | 
					    price = fields.Float(related='medicament_id.list_price', | 
				
			||||
 | 
					                         string="Price", | 
				
			||||
 | 
					                         help="Cost of medicine") | 
				
			||||
 | 
					    total = fields.Float(string="Total Price", | 
				
			||||
 | 
					                         help="Total price of medicine") | 
				
			||||
 | 
					    prescription_id = fields.Many2one('dental.prescription', | 
				
			||||
 | 
					                                      help="Relate the model with dental_prescription") | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					    @api.onchange('quantity') | 
				
			||||
 | 
					    def _onchange_quantity(self): | 
				
			||||
 | 
					        """Updates the total price of the medicament based on the quantity. | 
				
			||||
 | 
					        This method is triggered by an onchange event of the `quantity` field. | 
				
			||||
 | 
					        It calculates the total price by multiplying the `quantity` of the | 
				
			||||
 | 
					        medicament by its `price` and updates the `total` field with the new value. | 
				
			||||
 | 
					        """ | 
				
			||||
 | 
					        for rec in self: | 
				
			||||
 | 
					            rec.total = rec.price * rec.quantity | 
				
			||||
@ -0,0 +1,31 @@ | 
				
			|||||
 | 
					# -*- coding: utf-8 -*- | 
				
			||||
 | 
					################################################################################ | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    Cybrosys Technologies Pvt. Ltd. | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    Copyright (C) 2024-TODAY Cybrosys Technologies(<https://www.cybrosys.com>). | 
				
			||||
 | 
					#    Author: Cybrosys Techno Solutions(<https://www.cybrosys.com>) | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    You can modify it under the terms of the GNU AFFERO | 
				
			||||
 | 
					#    GENERAL PUBLIC LICENSE (AGPL v3), Version 3. | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    This program is distributed in the hope that it will be useful, | 
				
			||||
 | 
					#    but WITHOUT ANY WARRANTY; without even the implied warranty of | 
				
			||||
 | 
					#    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
				
			||||
 | 
					#    GNU AFFERO GENERAL PUBLIC LICENSE (AGPL v3) for more details. | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    You should have received a copy of the GNU AFFERO GENERAL PUBLIC LICENSE | 
				
			||||
 | 
					#    (AGPL v3) along with this program. | 
				
			||||
 | 
					#    If not, see <http://www.gnu.org/licenses/>. | 
				
			||||
 | 
					# | 
				
			||||
 | 
					################################################################################ | 
				
			||||
 | 
					from odoo import fields, models | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					class DentalSpecialist(models.Model): | 
				
			||||
 | 
					    """To mention doctors Specialised field""" | 
				
			||||
 | 
					    _name = 'dental.specialist' | 
				
			||||
 | 
					    _description = "Dental Specialist" | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					    name = fields.Char(string="Name", help="Name of the dental specialist") | 
				
			||||
 | 
					    code = fields.Char(string="Code", help="Add the code for the name") | 
				
			||||
@ -0,0 +1,53 @@ | 
				
			|||||
 | 
					# -*- coding: utf-8 -*- | 
				
			||||
 | 
					################################################################################ | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    Cybrosys Technologies Pvt. Ltd. | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    Copyright (C) 2024-TODAY Cybrosys Technologies(<https://www.cybrosys.com>). | 
				
			||||
 | 
					#    Author: Cybrosys Techno Solutions(<https://www.cybrosys.com>) | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    You can modify it under the terms of the GNU AFFERO | 
				
			||||
 | 
					#    GENERAL PUBLIC LICENSE (AGPL v3), Version 3. | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    This program is distributed in the hope that it will be useful, | 
				
			||||
 | 
					#    but WITHOUT ANY WARRANTY; without even the implied warranty of | 
				
			||||
 | 
					#    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
				
			||||
 | 
					#    GNU AFFERO GENERAL PUBLIC LICENSE (AGPL v3) for more details. | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    You should have received a copy of the GNU AFFERO GENERAL PUBLIC LICENSE | 
				
			||||
 | 
					#    (AGPL v3) along with this program. | 
				
			||||
 | 
					#    If not, see <http://www.gnu.org/licenses/>. | 
				
			||||
 | 
					# | 
				
			||||
 | 
					################################################################################ | 
				
			||||
 | 
					from odoo import api, fields, models | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					class DentalTimeShift(models.Model): | 
				
			||||
 | 
					    """Doctors time shift, different time slots""" | 
				
			||||
 | 
					    _name = 'dental.time.shift' | 
				
			||||
 | 
					    _description = "Dental Time Shift" | 
				
			||||
 | 
					    _rec_name = 'name' | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					    name = fields.Char(string='Name', readonly=True, | 
				
			||||
 | 
					                       help="name of the time shifts") | 
				
			||||
 | 
					    shift_type = fields.Selection( | 
				
			||||
 | 
					        selection=[('morning', 'Morning'), ('day', 'Day'), | 
				
			||||
 | 
					                   ('evening', 'Evening'), ('night', 'Night')], | 
				
			||||
 | 
					        string="Shift Type", help="Selection field for the shift type") | 
				
			||||
 | 
					    start_time = fields.Float(string="Start Time", help="start time of time slot") | 
				
			||||
 | 
					    end_time = fields.Float(string="End Time", help="End time of time slot") | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					    @api.model_create_multi | 
				
			||||
 | 
					    def create(self, vals_list): | 
				
			||||
 | 
					        """Overrides the default create method to set the `name` field of the | 
				
			||||
 | 
					        newly created `dental.time.shift` record(s) to a string that represents | 
				
			||||
 | 
					        the shift time range.""" | 
				
			||||
 | 
					        res = super(DentalTimeShift, self).create(vals_list) | 
				
			||||
 | 
					        res.name = f'{res.start_time} to {res.end_time}' | 
				
			||||
 | 
					        return res | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					    @api.onchange('start_time', 'end_time') | 
				
			||||
 | 
					    def _onchange_time(self): | 
				
			||||
 | 
					        name = f'{self.start_time} to {self.end_time}' | 
				
			||||
 | 
					        self.update({'name': name}) | 
				
			||||
 | 
					
 | 
				
			||||
@ -0,0 +1,36 @@ | 
				
			|||||
 | 
					# -*- coding: utf-8 -*- | 
				
			||||
 | 
					################################################################################ | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    Cybrosys Technologies Pvt. Ltd. | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    Copyright (C) 2024-TODAY Cybrosys Technologies(<https://www.cybrosys.com>). | 
				
			||||
 | 
					#    Author: Cybrosys Techno Solutions(<https://www.cybrosys.com>) | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    You can modify it under the terms of the GNU AFFERO | 
				
			||||
 | 
					#    GENERAL PUBLIC LICENSE (AGPL v3), Version 3. | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    This program is distributed in the hope that it will be useful, | 
				
			||||
 | 
					#    but WITHOUT ANY WARRANTY; without even the implied warranty of | 
				
			||||
 | 
					#    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
				
			||||
 | 
					#    GNU AFFERO GENERAL PUBLIC LICENSE (AGPL v3) for more details. | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    You should have received a copy of the GNU AFFERO GENERAL PUBLIC LICENSE | 
				
			||||
 | 
					#    (AGPL v3) along with this program. | 
				
			||||
 | 
					#    If not, see <http://www.gnu.org/licenses/>. | 
				
			||||
 | 
					# | 
				
			||||
 | 
					################################################################################ | 
				
			||||
 | 
					from odoo import fields, models | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					class DentalTreatment(models.Model): | 
				
			||||
 | 
					    """For adding Dental treatment details of the patients""" | 
				
			||||
 | 
					    _name = 'dental.treatment' | 
				
			||||
 | 
					    _description = "Dental Treatment" | 
				
			||||
 | 
					    _inherit = ['mail.thread'] | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					    name = fields.Char(string='Treatment Name', help="Date of the treatment") | 
				
			||||
 | 
					    treatment_categ_id = fields.Many2one('treatment.category', | 
				
			||||
 | 
					                                         string="Category", | 
				
			||||
 | 
					                                         help="name of the treatment") | 
				
			||||
 | 
					    cost = fields.Float(string='Cost', | 
				
			||||
 | 
					                        help="Cost of the Treatment") | 
				
			||||
@ -0,0 +1,32 @@ | 
				
			|||||
 | 
					# -*- coding: utf-8 -*- | 
				
			||||
 | 
					################################################################################ | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    Cybrosys Technologies Pvt. Ltd. | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    Copyright (C) 2024-TODAY Cybrosys Technologies(<https://www.cybrosys.com>). | 
				
			||||
 | 
					#    Author: Cybrosys Techno Solutions(<https://www.cybrosys.com>) | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    You can modify it under the terms of the GNU AFFERO | 
				
			||||
 | 
					#    GENERAL PUBLIC LICENSE (AGPL v3), Version 3. | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    This program is distributed in the hope that it will be useful, | 
				
			||||
 | 
					#    but WITHOUT ANY WARRANTY; without even the implied warranty of | 
				
			||||
 | 
					#    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
				
			||||
 | 
					#    GNU AFFERO GENERAL PUBLIC LICENSE (AGPL v3) for more details. | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    You should have received a copy of the GNU AFFERO GENERAL PUBLIC LICENSE | 
				
			||||
 | 
					#    (AGPL v3) along with this program. | 
				
			||||
 | 
					#    If not, see <http://www.gnu.org/licenses/>. | 
				
			||||
 | 
					# | 
				
			||||
 | 
					################################################################################ | 
				
			||||
 | 
					from odoo import fields, models | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					class InsuranceCompany(models.Model): | 
				
			||||
 | 
					    """To add the insurance details""" | 
				
			||||
 | 
					    _name = 'insurance.company' | 
				
			||||
 | 
					    _description = "Insurance Company" | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					    name = fields.Char(string="Name", help="Name of the insurance company") | 
				
			||||
 | 
					    phone = fields.Char(string="Phone", help="Phone number of the insurance company") | 
				
			||||
 | 
					    email = fields.Char(string="Email", help="Email of the insurance company") | 
				
			||||
@ -0,0 +1,38 @@ | 
				
			|||||
 | 
					# -*- coding: utf-8 -*- | 
				
			||||
 | 
					################################################################################ | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    Cybrosys Technologies Pvt. Ltd. | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    Copyright (C) 2024-TODAY Cybrosys Technologies(<https://www.cybrosys.com>). | 
				
			||||
 | 
					#    Author: Cybrosys Techno Solutions(<https://www.cybrosys.com>) | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    You can modify it under the terms of the GNU AFFERO | 
				
			||||
 | 
					#    GENERAL PUBLIC LICENSE (AGPL v3), Version 3. | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    This program is distributed in the hope that it will be useful, | 
				
			||||
 | 
					#    but WITHOUT ANY WARRANTY; without even the implied warranty of | 
				
			||||
 | 
					#    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
				
			||||
 | 
					#    GNU AFFERO GENERAL PUBLIC LICENSE (AGPL v3) for more details. | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    You should have received a copy of the GNU AFFERO GENERAL PUBLIC LICENSE | 
				
			||||
 | 
					#    (AGPL v3) along with this program. | 
				
			||||
 | 
					#    If not, see <http://www.gnu.org/licenses/>. | 
				
			||||
 | 
					# | 
				
			||||
 | 
					################################################################################ | 
				
			||||
 | 
					from odoo import fields, models | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					class MedicalQuestionnaire(models.Model): | 
				
			||||
 | 
					    """Medical questions to be asked to the patients while their appointment""" | 
				
			||||
 | 
					    _name = 'medical.questionnaire' | 
				
			||||
 | 
					    _description = 'Medical Questionnaire' | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					    question_id = fields.Many2one('medical.questions', | 
				
			||||
 | 
					                                  string='Questions', | 
				
			||||
 | 
					                                  help="All added question") | 
				
			||||
 | 
					    yes_no = fields.Selection([('yes', 'Yes'), ('no', 'No')], | 
				
			||||
 | 
					                              string='Yes or No', help="") | 
				
			||||
 | 
					    reason = fields.Text(string='Reason', help="Reason for the question answer") | 
				
			||||
 | 
					    patient_id = fields.Many2one('res.partner', | 
				
			||||
 | 
					                                 string='Patient', | 
				
			||||
 | 
					                                 help="Patient name") | 
				
			||||
@ -0,0 +1,51 @@ | 
				
			|||||
 | 
					# -*- coding: utf-8 -*- | 
				
			||||
 | 
					################################################################################ | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    Cybrosys Technologies Pvt. Ltd. | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    Copyright (C) 2024-TODAY Cybrosys Technologies(<https://www.cybrosys.com>). | 
				
			||||
 | 
					#    Author: Cybrosys Techno Solutions(<https://www.cybrosys.com>) | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    You can modify it under the terms of the GNU AFFERO | 
				
			||||
 | 
					#    GENERAL PUBLIC LICENSE (AGPL v3), Version 3. | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    This program is distributed in the hope that it will be useful, | 
				
			||||
 | 
					#    but WITHOUT ANY WARRANTY; without even the implied warranty of | 
				
			||||
 | 
					#    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
				
			||||
 | 
					#    GNU AFFERO GENERAL PUBLIC LICENSE (AGPL v3) for more details. | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    You should have received a copy of the GNU AFFERO GENERAL PUBLIC LICENSE | 
				
			||||
 | 
					#    (AGPL v3) along with this program. | 
				
			||||
 | 
					#    If not, see <http://www.gnu.org/licenses/>. | 
				
			||||
 | 
					# | 
				
			||||
 | 
					################################################################################ | 
				
			||||
 | 
					from odoo import api, fields, models | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					class MedicalQuestions(models.Model): | 
				
			||||
 | 
					    """To add medical questionnaire question""" | 
				
			||||
 | 
					    _name = 'medical.questions' | 
				
			||||
 | 
					    _description = 'Medical Questions' | 
				
			||||
 | 
					    _rec_name = 'question' | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					    question = fields.Char(string='Question') | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					    @api.model | 
				
			||||
 | 
					    def create(self, vals): | 
				
			||||
 | 
					        """Overrides the default create method to add a new medical question | 
				
			||||
 | 
					        record and automatically create a corresponding entry in the | 
				
			||||
 | 
					        `medical.questionnaire` model.""" | 
				
			||||
 | 
					        res = super(MedicalQuestions, self).create(vals) | 
				
			||||
 | 
					        self.env['medical.questionnaire'].create({ | 
				
			||||
 | 
					            'question_id': res.id | 
				
			||||
 | 
					        }) | 
				
			||||
 | 
					        return res | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					    def unlink(self): | 
				
			||||
 | 
					        """Overrides the default unlink method to delete the current medical question record. | 
				
			||||
 | 
					        Before deletion, it searches for and deletes any associated records in the | 
				
			||||
 | 
					        `medical.questionnaire` model that reference this medical question.""" | 
				
			||||
 | 
					        for rec in self: | 
				
			||||
 | 
					            for line in self.env['medical.questionnaire'].search([('question_id', '=', rec.id)]): | 
				
			||||
 | 
					                line.unlink() | 
				
			||||
 | 
					            return super(MedicalQuestions, self).unlink() | 
				
			||||
@ -0,0 +1,33 @@ | 
				
			|||||
 | 
					# -*- coding: utf-8 -*- | 
				
			||||
 | 
					################################################################################ | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    Cybrosys Technologies Pvt. Ltd. | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    Copyright (C) 2024-TODAY Cybrosys Technologies(<https://www.cybrosys.com>). | 
				
			||||
 | 
					#    Author: Cybrosys Techno Solutions(<https://www.cybrosys.com>) | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    You can modify it under the terms of the GNU AFFERO | 
				
			||||
 | 
					#    GENERAL PUBLIC LICENSE (AGPL v3), Version 3. | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    This program is distributed in the hope that it will be useful, | 
				
			||||
 | 
					#    but WITHOUT ANY WARRANTY; without even the implied warranty of | 
				
			||||
 | 
					#    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
				
			||||
 | 
					#    GNU AFFERO GENERAL PUBLIC LICENSE (AGPL v3) for more details. | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    You should have received a copy of the GNU AFFERO GENERAL PUBLIC LICENSE | 
				
			||||
 | 
					#    (AGPL v3) along with this program. | 
				
			||||
 | 
					#    If not, see <http://www.gnu.org/licenses/>. | 
				
			||||
 | 
					# | 
				
			||||
 | 
					################################################################################ | 
				
			||||
 | 
					from odoo import fields, models | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					class MedicineFrequency(models.Model): | 
				
			||||
 | 
					    """To specifing the medicine frequency, how to consume it.""" | 
				
			||||
 | 
					    _name = 'medicine.frequency' | 
				
			||||
 | 
					    _description = "Medicine Frequency" | 
				
			||||
 | 
					    _rec_name = "medicament_frequency" | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					    code = fields.Char(string="Code", help="code of medicine frequency") | 
				
			||||
 | 
					    medicament_frequency = fields.Char(string="Medicine Frequency", | 
				
			||||
 | 
					                                       help="Add the frquency of medicine how to eat") | 
				
			||||
@ -0,0 +1,30 @@ | 
				
			|||||
 | 
					# -*- coding: utf-8 -*- | 
				
			||||
 | 
					################################################################################ | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    Cybrosys Technologies Pvt. Ltd. | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    Copyright (C) 2024-TODAY Cybrosys Technologies(<https://www.cybrosys.com>). | 
				
			||||
 | 
					#    Author: Cybrosys Techno Solutions(<https://www.cybrosys.com>) | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    You can modify it under the terms of the GNU AFFERO | 
				
			||||
 | 
					#    GENERAL PUBLIC LICENSE (AGPL v3), Version 3. | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    This program is distributed in the hope that it will be useful, | 
				
			||||
 | 
					#    but WITHOUT ANY WARRANTY; without even the implied warranty of | 
				
			||||
 | 
					#    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
				
			||||
 | 
					#    GNU AFFERO GENERAL PUBLIC LICENSE (AGPL v3) for more details. | 
				
			||||
 | 
					# | 
				
			||||
 | 
					#    You should have received a copy of the GNU AFFERO GENERAL PUBLIC LICENSE | 
				
			||||
 | 
					#    (AGPL v3) along with this program. | 
				
			||||
 | 
					#    If not, see <http://www.gnu.org/licenses/>. | 
				
			||||
 | 
					# | 
				
			||||
 | 
					################################################################################ | 
				
			||||
 | 
					from odoo import fields, models | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					class TreatmentCategory(models.Model): | 
				
			||||
 | 
					    """Adding the treatment category""" | 
				
			||||
 | 
					    _name = 'treatment.category' | 
				
			||||
 | 
					    _description = "Treatment Category" | 
				
			||||
 | 
					
 | 
				
			||||
 | 
					    name = fields.Char(string="Name", help="Name of the treatment category") | 
				
			||||
@ -0,0 +1,14 @@ | 
				
			|||||
 | 
					<?xml version="1.0" encoding="utf-8"?> | 
				
			||||
 | 
					<odoo> | 
				
			||||
 | 
					    <!-- Report Action of Dental Prescription Pdf --> | 
				
			||||
 | 
					    <record id="report_pdf_dental_prescription" model="ir.actions.report"> | 
				
			||||
 | 
					        <field name="name">Prescription Pdf</field> | 
				
			||||
 | 
					        <field name="model">dental.prescription</field> | 
				
			||||
 | 
					        <field name="report_type">qweb-pdf</field> | 
				
			||||
 | 
					        <field name="report_name">dental_clinic_management.report_prescription</field> | 
				
			||||
 | 
					        <field name="report_file">dental_clinic_management.report_prescription</field> | 
				
			||||
 | 
					        <field name="print_report_name">'Prescription Report - %s' % (object.sequence_no)</field> | 
				
			||||
 | 
					        <field name="binding_model_id" ref="dental_clinic_management.model_dental_prescription"/> | 
				
			||||
 | 
					        <field name="binding_type">report</field> | 
				
			||||
 | 
					    </record> | 
				
			||||
 | 
					</odoo> | 
				
			||||
@ -0,0 +1,119 @@ | 
				
			|||||
 | 
					<?xml version="1.0" encoding="utf-8"?> | 
				
			||||
 | 
					<odoo> | 
				
			||||
 | 
					<!-- Pdf template of Prescription Report--> | 
				
			||||
 | 
					    <template id="report_prescription"> | 
				
			||||
 | 
					        <t t-call="web.html_container"> | 
				
			||||
 | 
					            <t t-foreach="docs" t-as="o"> | 
				
			||||
 | 
					                <t t-call="web.external_layout"> | 
				
			||||
 | 
					                    <div class="page"> | 
				
			||||
 | 
					                        <div class="oe_structure"/> | 
				
			||||
 | 
					                        <div class="text-center"> | 
				
			||||
 | 
					                            <h2>DENTAL PRESCRIPTION REPORT</h2> | 
				
			||||
 | 
					                        </div> | 
				
			||||
 | 
					                        <br></br> | 
				
			||||
 | 
					                        <table class="table table-borderless"> | 
				
			||||
 | 
					                            <tbody> | 
				
			||||
 | 
					                                <tr> | 
				
			||||
 | 
					                                    <td> | 
				
			||||
 | 
					                                        Patient Name : | 
				
			||||
 | 
					                                        <span t-field="o.patient_id"/> | 
				
			||||
 | 
					                                    </td> | 
				
			||||
 | 
					                                    <td> | 
				
			||||
 | 
					                                        Doctor Name : | 
				
			||||
 | 
					                                        <span t-field="o.prescribed_doctor_id"/> | 
				
			||||
 | 
					                                    </td> | 
				
			||||
 | 
					                                </tr> | 
				
			||||
 | 
					                                <tr> | 
				
			||||
 | 
					                                    <td> | 
				
			||||
 | 
					                                        Appointment ID : | 
				
			||||
 | 
					                                        <span t-field="o.appointment_id"/> | 
				
			||||
 | 
					                                    </td> | 
				
			||||
 | 
					                                    <td> | 
				
			||||
 | 
					                                        Treatment : | 
				
			||||
 | 
					                                        <span t-field="o.treatment_id"/> | 
				
			||||
 | 
					                                    </td> | 
				
			||||
 | 
					                                </tr> | 
				
			||||
 | 
					                                <tr> | 
				
			||||
 | 
					                                    <td> | 
				
			||||
 | 
					                                        Token NO : | 
				
			||||
 | 
					                                        <span t-field="o.token_no"/> | 
				
			||||
 | 
					                                    </td> | 
				
			||||
 | 
					                                    <td> | 
				
			||||
 | 
					                                        Date : | 
				
			||||
 | 
					                                        <span t-field="o.prescription_date"/> | 
				
			||||
 | 
					                                    </td> | 
				
			||||
 | 
					                                </tr> | 
				
			||||
 | 
					                            </tbody> | 
				
			||||
 | 
					                        </table> | 
				
			||||
 | 
					                        <table class="table table-bordered mt32"> | 
				
			||||
 | 
					                            <thead> | 
				
			||||
 | 
					                                <tr> | 
				
			||||
 | 
					                                    <th>Sl.No</th> | 
				
			||||
 | 
					                                    <t t-set="counter" t-value="0"/> | 
				
			||||
 | 
					                                    <th class="text-center"> | 
				
			||||
 | 
					                                        <span>Medicament</span> | 
				
			||||
 | 
					                                    </th> | 
				
			||||
 | 
					                                    <th class="text-center"> | 
				
			||||
 | 
					                                        <span>Generic Name</span> | 
				
			||||
 | 
					                                    </th> | 
				
			||||
 | 
					                                    <th class="text-center"> | 
				
			||||
 | 
					                                        <span>Dosage Strength</span> | 
				
			||||
 | 
					                                    </th> | 
				
			||||
 | 
					                                    <th class="text-center"> | 
				
			||||
 | 
					                                        <span>Medicament Form</span> | 
				
			||||
 | 
					                                    </th> | 
				
			||||
 | 
					                                    <th class="text-center"> | 
				
			||||
 | 
					                                        <span>Quantity</span> | 
				
			||||
 | 
					                                    </th> | 
				
			||||
 | 
					                                    <th class="text-center"> | 
				
			||||
 | 
					                                        <span>Frequency</span> | 
				
			||||
 | 
					                                    </th> | 
				
			||||
 | 
					                                </tr> | 
				
			||||
 | 
					                            </thead> | 
				
			||||
 | 
					                            <tbody> | 
				
			||||
 | 
					                                <t t-foreach="o.medicine_ids" t-as="medicine"> | 
				
			||||
 | 
					                                    <tr class="text-center"> | 
				
			||||
 | 
					                                        <td style="text-align:center;"> | 
				
			||||
 | 
					                                            <t t-set="counter" t-value="counter + 1"/> | 
				
			||||
 | 
					                                            <t t-esc="counter"/> | 
				
			||||
 | 
					                                        </td> | 
				
			||||
 | 
					                                        <td style="text-align:center;"> | 
				
			||||
 | 
					                                            <span> | 
				
			||||
 | 
					                                                <t t-esc="medicine.medicament_id.name"/> | 
				
			||||
 | 
					                                            </span> | 
				
			||||
 | 
					                                        </td> | 
				
			||||
 | 
					                                        <td style="text-align:center;"> | 
				
			||||
 | 
					                                            <span> | 
				
			||||
 | 
					                                                <t t-esc="medicine.generic_name"/> | 
				
			||||
 | 
					                                            </span> | 
				
			||||
 | 
					                                        </td> | 
				
			||||
 | 
					                                        <td style="text-align:center;"> | 
				
			||||
 | 
					                                            <span> | 
				
			||||
 | 
					                                                <t t-esc="medicine.dosage_strength"/> | 
				
			||||
 | 
					                                            </span> | 
				
			||||
 | 
					                                        </td> | 
				
			||||
 | 
					                                        <td style="text-align:center;"> | 
				
			||||
 | 
					                                            <span> | 
				
			||||
 | 
					                                                <t t-esc="medicine.medicament_form"/> | 
				
			||||
 | 
					                                            </span> | 
				
			||||
 | 
					                                        </td> | 
				
			||||
 | 
					                                        <td style="text-align:center;"> | 
				
			||||
 | 
					                                            <span> | 
				
			||||
 | 
					                                                <t t-esc="medicine.quantity"/> | 
				
			||||
 | 
					                                            </span> | 
				
			||||
 | 
					                                        </td> | 
				
			||||
 | 
					                                        <td style="text-align:center;"> | 
				
			||||
 | 
					                                            <span> | 
				
			||||
 | 
					                                                <t t-esc="medicine.frequency_id.medicament_frequency"/> | 
				
			||||
 | 
					                                            </span> | 
				
			||||
 | 
					                                        </td> | 
				
			||||
 | 
					                                    </tr> | 
				
			||||
 | 
					                                </t> | 
				
			||||
 | 
					                            </tbody> | 
				
			||||
 | 
					                        </table> | 
				
			||||
 | 
					                    </div> | 
				
			||||
 | 
					                </t> | 
				
			||||
 | 
					            </t> | 
				
			||||
 | 
					        </t> | 
				
			||||
 | 
					    </template> | 
				
			||||
 | 
					</odoo> | 
				
			||||
@ -0,0 +1,23 @@ | 
				
			|||||
 | 
					<?xml version="1.0" encoding="utf-8"?> | 
				
			||||
 | 
					<odoo> | 
				
			||||
 | 
					<!-- Created the category Dental Clinic to set the Groups --> | 
				
			||||
 | 
					    <record id="module_category_dental_clinic" model="ir.module.category"> | 
				
			||||
 | 
					        <field name="name">Dental Clinic</field> | 
				
			||||
 | 
					        <field name="sequence">18</field> | 
				
			||||
 | 
					    </record> | 
				
			||||
 | 
					<!-- Created the Groups as Own Documents Only and this groups will be only access their own datas --> | 
				
			||||
 | 
					    <record id="group_dental_doctor" model="res.groups"> | 
				
			||||
 | 
					        <field name="name">User: Own Documents Only</field> | 
				
			||||
 | 
					        <field name="category_id" ref="module_category_dental_clinic"/> | 
				
			||||
 | 
					        <field name="implied_ids" eval="[(4, ref('base.group_user'))]"/> | 
				
			||||
 | 
					        <field name="comment">the user will have access to his own data in the dental clinic application.</field> | 
				
			||||
 | 
					    </record> | 
				
			||||
 | 
					<!-- Created the Groups as OManager and this groups can we view all the datas --> | 
				
			||||
 | 
					    <record id="group_dental_manager" model="res.groups"> | 
				
			||||
 | 
					        <field name="name">Manager</field> | 
				
			||||
 | 
					        <field name="comment">the user will have an access to all the dental clinic configuration.</field> | 
				
			||||
 | 
					        <field name="category_id" ref="module_category_dental_clinic"/> | 
				
			||||
 | 
					        <field name="implied_ids" eval="[(4, ref('group_dental_doctor'))]"/> | 
				
			||||
 | 
					        <field name="users" eval="[(4, ref('base.user_root'))]"/> | 
				
			||||
 | 
					    </record> | 
				
			||||
 | 
					</odoo> | 
				
			||||
@ -0,0 +1,33 @@ | 
				
			|||||
 | 
					<?xml version="1.0" encoding="utf-8"?> | 
				
			||||
 | 
					<odoo> | 
				
			||||
 | 
					    <!-- Set the record rule for visible only the corresponding doctor appointments --> | 
				
			||||
 | 
					    <record id="own_dental_appointments" model="ir.rule"> | 
				
			||||
 | 
					        <field name="name">Own Appointments</field> | 
				
			||||
 | 
					        <field name="model_id" ref="model_dental_appointment"/> | 
				
			||||
 | 
					        <field name="domain_force">[('doctor_id.user_id', '=', user.id)]</field> | 
				
			||||
 | 
					        <field name="groups" | 
				
			||||
 | 
					               eval="[Command.link(ref('dental_clinic_management.group_dental_doctor'))]"/> | 
				
			||||
 | 
					    </record> | 
				
			||||
 | 
					    <!-- To see all the appointments for the manager --> | 
				
			||||
 | 
					    <record id="see_all_dental_appointments" model="ir.rule"> | 
				
			||||
 | 
					        <field name="name">All Appointments</field> | 
				
			||||
 | 
					        <field ref="model_dental_appointment" name="model_id"/> | 
				
			||||
 | 
					        <field name="domain_force">[(1, '=', 1)]</field> | 
				
			||||
 | 
					        <field name="groups" eval="[Command.link(ref('dental_clinic_management.group_dental_manager'))]"/> | 
				
			||||
 | 
					    </record> | 
				
			||||
 | 
					    <!-- Set the record rule for visible only the corresponding doctor prescriptions --> | 
				
			||||
 | 
					    <record id="own_dental_prescription" model="ir.rule"> | 
				
			||||
 | 
					        <field name="name">Own Prescriptions</field> | 
				
			||||
 | 
					        <field name="model_id" ref="model_dental_prescription"/> | 
				
			||||
 | 
					        <field name="domain_force">[('prescribed_doctor_id.user_id', '=', user.id)]</field> | 
				
			||||
 | 
					        <field name="groups" | 
				
			||||
 | 
					               eval="[Command.link(ref('dental_clinic_management.group_dental_doctor'))]"/> | 
				
			||||
 | 
					    </record> | 
				
			||||
 | 
					    <!-- To see all the prescriptions for the manager --> | 
				
			||||
 | 
					    <record id="see_all_dental_prescriptions" model="ir.rule"> | 
				
			||||
 | 
					        <field name="name">All Prescriptions</field> | 
				
			||||
 | 
					        <field ref="model_dental_prescription" name="model_id"/> | 
				
			||||
 | 
					        <field name="domain_force">[(1, '=', 1)]</field> | 
				
			||||
 | 
					        <field name="groups" eval="[Command.link(ref('dental_clinic_management.group_dental_manager'))]"/> | 
				
			||||
 | 
					    </record> | 
				
			||||
 | 
					</odoo> | 
				
			||||
		
		
			
  | 
| 
		 After Width: | Height: | Size: 36 KiB  | 
| 
		 After Width: | Height: | Size: 3.6 KiB  | 
| 
		 After Width: | Height: | Size: 310 B  | 
| 
		 After Width: | Height: | Size: 1.3 KiB  | 
| 
		 After Width: | Height: | Size: 1.4 KiB  | 
| 
		 After Width: | Height: | Size: 576 B  | 
| 
		 After Width: | Height: | Size: 733 B  | 
| 
		 After Width: | Height: | Size: 911 B  | 
| 
		 After Width: | Height: | Size: 1.1 KiB  | 
| 
		 After Width: | Height: | Size: 1.1 KiB  | 
| 
		 After Width: | Height: | Size: 1.2 KiB  | 
| 
		 After Width: | Height: | Size: 673 B  | 
| 
		 After Width: | Height: | Size: 11 KiB  | 
| 
		 After Width: | Height: | Size: 878 B  | 
| 
		 After Width: | Height: | Size: 653 B  | 
| 
		 After Width: | Height: | Size: 905 B  | 
| 
		 After Width: | Height: | Size: 839 B  | 
| 
		 After Width: | Height: | Size: 427 B  | 
| 
		 After Width: | Height: | Size: 627 B  | 
| 
		 After Width: | Height: | Size: 1.2 KiB  | 
| 
		 After Width: | Height: | Size: 988 B  | 
| 
		 After Width: | Height: | Size: 1.2 KiB  | 
| 
		 After Width: | Height: | Size: 80 KiB  | 
| 
		 After Width: | Height: | Size: 3.2 KiB  | 
| 
		 After Width: | Height: | Size: 565 B  | 
| 
		 After Width: | Height: | Size: 26 KiB  | 
| 
		 After Width: | Height: | Size: 43 KiB  | 
| 
		 After Width: | Height: | Size: 1.4 KiB  | 
| 
		 After Width: | Height: | Size: 4.0 KiB  | 
| 
		 After Width: | Height: | Size: 38 KiB  | 
| 
		 After Width: | Height: | Size: 4.3 KiB  | 
| 
		 After Width: | Height: | Size: 87 KiB  | 
| 
		 After Width: | Height: | Size: 80 KiB  | 
| 
		 After Width: | Height: | Size: 95 KiB  | 
| 
		 After Width: | Height: | Size: 78 KiB  | 
| 
		 After Width: | Height: | Size: 164 KiB  | 
| 
		 After Width: | Height: | Size: 81 KiB  | 
| 
		 After Width: | Height: | Size: 80 KiB  | 
| 
		 After Width: | Height: | Size: 60 KiB  | 
| 
		 After Width: | Height: | Size: 36 KiB  | 
| 
		 After Width: | Height: | Size: 97 KiB  | 
| 
		 After Width: | Height: | Size: 122 KiB  | 
| 
		 After Width: | Height: | Size: 96 KiB  | 
| 
		 After Width: | Height: | Size: 86 KiB  | 
| 
		 After Width: | Height: | Size: 102 KiB  | 
| 
		 After Width: | Height: | Size: 102 KiB  | 
| 
		 After Width: | Height: | Size: 61 KiB  | 
| 
		 After Width: | Height: | Size: 38 KiB  | 
| 
		 After Width: | Height: | Size: 52 KiB  | 
| 
		 After Width: | Height: | Size: 90 KiB  | 
| 
		 After Width: | Height: | Size: 70 KiB  | 
| 
		 After Width: | Height: | Size: 71 KiB  | 
| 
		 After Width: | Height: | Size: 48 KiB  | 
| 
		 After Width: | Height: | Size: 46 KiB  | 
| 
		 After Width: | Height: | Size: 115 KiB  | 
| 
		 After Width: | Height: | Size: 40 KiB  | 
| 
		 After Width: | Height: | Size: 48 KiB  | 
| 
		 After Width: | Height: | Size: 80 KiB  | 
| 
		 After Width: | Height: | Size: 32 KiB  | 
| 
		 After Width: | Height: | Size: 101 KiB  | 
| 
		 After Width: | Height: | Size: 50 KiB  | 
| 
		 After Width: | Height: | Size: 41 KiB  | 
| 
		 After Width: | Height: | Size: 132 KiB  | 
| 
		 After Width: | Height: | Size: 93 KiB  | 
| 
		 After Width: | Height: | Size: 107 KiB  |