Browse Source

Add dental-clinic-management to Odoo 18

pull/357/head
Ruby Hartono 5 months ago
parent
commit
4a94bf8bdf
  1. 48
      dental_clinical_management/README.rst
  2. 24
      dental_clinical_management/__init__.py
  3. 79
      dental_clinical_management/__manifest__.py
  4. 23
      dental_clinical_management/controllers/__init__.py
  5. 124
      dental_clinical_management/controllers/dental_clinic.py
  6. 65
      dental_clinical_management/controllers/patient_portal.py
  7. 15
      dental_clinical_management/data/dental_department_data.xml
  8. 18
      dental_clinical_management/data/dental_specialist_data.xml
  9. 26
      dental_clinical_management/data/dental_time_shift_data.xml
  10. 26
      dental_clinical_management/data/dental_treatment_data.xml
  11. 15
      dental_clinical_management/data/insurance_company_data.xml
  12. 19
      dental_clinical_management/data/ir_sequence.xml
  13. 30
      dental_clinical_management/data/medicine_frequency_data.xml
  14. 15
      dental_clinical_management/data/treatment_category_data.xml
  15. 17
      dental_clinical_management/data/website_menu.xml
  16. 6
      dental_clinical_management/doc/RELEASE_NOTES.md
  17. 34
      dental_clinical_management/models/__init__.py
  18. 131
      dental_clinical_management/models/dental_appointment.py
  19. 65
      dental_clinical_management/models/dental_doctor.py
  20. 36
      dental_clinical_management/models/dental_medicine.py
  21. 121
      dental_clinical_management/models/dental_patients.py
  22. 221
      dental_clinical_management/models/dental_prescription.py
  23. 31
      dental_clinical_management/models/dental_specialist.py
  24. 53
      dental_clinical_management/models/dental_time_shift.py
  25. 36
      dental_clinical_management/models/dental_treatment.py
  26. 32
      dental_clinical_management/models/insurance_company.py
  27. 38
      dental_clinical_management/models/medical_questionnaire.py
  28. 51
      dental_clinical_management/models/medical_questions.py
  29. 33
      dental_clinical_management/models/medicine_frequency.py
  30. 30
      dental_clinical_management/models/treatment_category.py
  31. 64
      dental_clinical_management/report/dental_appointment_card.xml
  32. 14
      dental_clinical_management/report/dental_prescription_report.xml
  33. 119
      dental_clinical_management/report/dental_prescription_templates.xml
  34. 23
      dental_clinical_management/security/dental_clinical_management_groups.xml
  35. 33
      dental_clinical_management/security/dental_clinical_management_security.xml
  36. 27
      dental_clinical_management/security/ir.model.access.csv
  37. BIN
      dental_clinical_management/static/description/assets/icons/capture (1).png
  38. BIN
      dental_clinical_management/static/description/assets/icons/check.png
  39. BIN
      dental_clinical_management/static/description/assets/icons/chevron.png
  40. BIN
      dental_clinical_management/static/description/assets/icons/cogs.png
  41. BIN
      dental_clinical_management/static/description/assets/icons/consultation.png
  42. BIN
      dental_clinical_management/static/description/assets/icons/ecom-black.png
  43. BIN
      dental_clinical_management/static/description/assets/icons/education-black.png
  44. BIN
      dental_clinical_management/static/description/assets/icons/hotel-black.png
  45. BIN
      dental_clinical_management/static/description/assets/icons/img.png
  46. BIN
      dental_clinical_management/static/description/assets/icons/license.png
  47. BIN
      dental_clinical_management/static/description/assets/icons/lifebuoy.png
  48. BIN
      dental_clinical_management/static/description/assets/icons/manufacturing-black.png
  49. BIN
      dental_clinical_management/static/description/assets/icons/photo-capture.png
  50. BIN
      dental_clinical_management/static/description/assets/icons/pos-black.png
  51. BIN
      dental_clinical_management/static/description/assets/icons/puzzle.png
  52. BIN
      dental_clinical_management/static/description/assets/icons/restaurant-black.png
  53. BIN
      dental_clinical_management/static/description/assets/icons/service-black.png
  54. BIN
      dental_clinical_management/static/description/assets/icons/trading-black.png
  55. BIN
      dental_clinical_management/static/description/assets/icons/training.png
  56. BIN
      dental_clinical_management/static/description/assets/icons/update.png
  57. BIN
      dental_clinical_management/static/description/assets/icons/user.png
  58. BIN
      dental_clinical_management/static/description/assets/icons/wrench.png
  59. BIN
      dental_clinical_management/static/description/assets/misc/Cybrosys R.png
  60. 33
      dental_clinical_management/static/description/assets/misc/email.svg
  61. 3
      dental_clinical_management/static/description/assets/misc/phone.svg
  62. 9
      dental_clinical_management/static/description/assets/misc/star (1) 2.svg
  63. 9
      dental_clinical_management/static/description/assets/misc/support (1) 1.svg
  64. 6
      dental_clinical_management/static/description/assets/misc/support-email.svg
  65. 17
      dental_clinical_management/static/description/assets/misc/tick-mark.svg
  66. 9
      dental_clinical_management/static/description/assets/misc/whatsapp 1.svg
  67. 33
      dental_clinical_management/static/description/assets/misc/whatsapp.svg
  68. BIN
      dental_clinical_management/static/description/assets/modules/1.jpg
  69. BIN
      dental_clinical_management/static/description/assets/modules/2.png
  70. BIN
      dental_clinical_management/static/description/assets/modules/3.jpg
  71. BIN
      dental_clinical_management/static/description/assets/modules/4.jpg
  72. BIN
      dental_clinical_management/static/description/assets/modules/5.jpg
  73. BIN
      dental_clinical_management/static/description/assets/modules/6.jpg
  74. BIN
      dental_clinical_management/static/description/assets/screenshots/10.png
  75. BIN
      dental_clinical_management/static/description/assets/screenshots/11.png
  76. BIN
      dental_clinical_management/static/description/assets/screenshots/12.png
  77. BIN
      dental_clinical_management/static/description/assets/screenshots/13.png
  78. BIN
      dental_clinical_management/static/description/assets/screenshots/14.png
  79. BIN
      dental_clinical_management/static/description/assets/screenshots/15.png
  80. BIN
      dental_clinical_management/static/description/assets/screenshots/16.png
  81. BIN
      dental_clinical_management/static/description/assets/screenshots/17.png
  82. BIN
      dental_clinical_management/static/description/assets/screenshots/18.png
  83. BIN
      dental_clinical_management/static/description/assets/screenshots/19.png
  84. BIN
      dental_clinical_management/static/description/assets/screenshots/2.png
  85. BIN
      dental_clinical_management/static/description/assets/screenshots/20.png
  86. BIN
      dental_clinical_management/static/description/assets/screenshots/21.png
  87. BIN
      dental_clinical_management/static/description/assets/screenshots/22.png
  88. BIN
      dental_clinical_management/static/description/assets/screenshots/23.png
  89. BIN
      dental_clinical_management/static/description/assets/screenshots/24.png
  90. BIN
      dental_clinical_management/static/description/assets/screenshots/25.png
  91. BIN
      dental_clinical_management/static/description/assets/screenshots/26.png
  92. BIN
      dental_clinical_management/static/description/assets/screenshots/27.png
  93. BIN
      dental_clinical_management/static/description/assets/screenshots/28.png
  94. BIN
      dental_clinical_management/static/description/assets/screenshots/29.png
  95. BIN
      dental_clinical_management/static/description/assets/screenshots/3.png
  96. BIN
      dental_clinical_management/static/description/assets/screenshots/30.png
  97. BIN
      dental_clinical_management/static/description/assets/screenshots/31.png
  98. BIN
      dental_clinical_management/static/description/assets/screenshots/32.png
  99. BIN
      dental_clinical_management/static/description/assets/screenshots/33.png
  100. BIN
      dental_clinical_management/static/description/assets/screenshots/34.png

48
dental_clinical_management/README.rst

@ -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>`__

24
dental_clinical_management/__init__.py

@ -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

79
dental_clinical_management/__manifest__.py

@ -0,0 +1,79 @@
# -*- 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': '18.0.1.0.0',
'category': 'Services',
'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_clinical_management/static/src/js/dental_clinic.js"
]
},
'data': [
'security/dental_clinical_management_groups.xml',
'security/dental_clinical_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_appointment_card.xml',
'report/dental_prescription_templates.xml',
'report/dental_prescription_report.xml',
'wizard/xray_report_views.xml',
'views/dental_clinical_management_menu.xml',
],
'images': ['static/description/banner.jpg'],
'license': 'AGPL-3',
'installable': True,
'auto_install': False,
'application': True,
}

23
dental_clinical_management/controllers/__init__.py

@ -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

124
dental_clinical_management/controllers/dental_clinic.py

@ -0,0 +1,124 @@
# -*- 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 io import BytesIO
from reportlab.pdfgen import canvas
from odoo.exceptions import UserError
from odoo import _, http
from odoo.http import Controller, request, route, content_disposition
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_clinical_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(self.env._('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_clinical_management.website_rental_success_template', {'token': kwargs})
@http.route('/dental_clinic/appointment_card/<int:token>',
type='http', auth="public", website=True)
def appointment_card(self, token):
"""To download the appointment card for patients for doctor's appointment"""
appointment = request.env['dental.appointment'].sudo().browse(token)
if not appointment.exists():
return request.not_found()
data = {
'token': appointment.token_no,
'doctor': appointment.doctor_id.name,
'specialised': appointment.specialist_id.name,
'appointment_time': appointment.shift_id.name,
'date': appointment.date,
}
report_service = request.env['ir.actions.report']
pdf_content, _ = report_service._render_qweb_pdf(
'dental_clinical_management.action_appointment_card', data=data)
pdf_http_headers = [('Content-Type', 'application/pdf'),
('Content-Length', len(pdf_content))]
return request.make_response(pdf_content, headers=pdf_http_headers)
@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_clinical_management.website_all_doctors',
{'doctor_ids': doctor_id})

65
dental_clinical_management/controllers/patient_portal.py

@ -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_clinical_management.group_dental_manager') in request.env.user.groups_id:
domain = []
elif request.env.ref('dental_clinical_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_clinical_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_clinical_management.prescription_portal_template',
{'prescription_details': prescription, 'page_name': 'prescription'})

15
dental_clinical_management/data/dental_department_data.xml

@ -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>

18
dental_clinical_management/data/dental_specialist_data.xml

@ -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>

26
dental_clinical_management/data/dental_time_shift_data.xml

@ -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>

26
dental_clinical_management/data/dental_treatment_data.xml

@ -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>

15
dental_clinical_management/data/insurance_company_data.xml

@ -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>

19
dental_clinical_management/data/ir_sequence.xml

@ -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>

30
dental_clinical_management/data/medicine_frequency_data.xml

@ -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>

15
dental_clinical_management/data/treatment_category_data.xml

@ -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>

17
dental_clinical_management/data/website_menu.xml

@ -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>

6
dental_clinical_management/doc/RELEASE_NOTES.md

@ -0,0 +1,6 @@
## Module <dental_clinical_management>
#### 05.07.2024
#### Version 17.0.1.0.0
#### ADD
Initial commit for Dental Clinic Management

34
dental_clinical_management/models/__init__.py

@ -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

131
dental_clinical_management/models/dental_appointment.py

@ -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: self.env._('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', self.env._('New')) == self.env._('New'):
vals['sequence_no'] = self.env['ir.sequence'].next_by_code(
'dental.appointment') or self.env._('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,
}

65
dental_clinical_management/models/dental_doctor.py

@ -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

36
dental_clinical_management/models/dental_medicine.py

@ -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")

121
dental_clinical_management/models/dental_patients.py

@ -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_clinical_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(self.env._("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

221
dental_clinical_management/models/dental_prescription.py

@ -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: self.env._('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', self.env._('New')) == self.env._('New'):
vals['sequence_no'] = self.env['ir.sequence'].next_by_code(
'dental.prescriptions') or self.env._('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': self.env._('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': self.env._('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

31
dental_clinical_management/models/dental_specialist.py

@ -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")

53
dental_clinical_management/models/dental_time_shift.py

@ -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})

36
dental_clinical_management/models/dental_treatment.py

@ -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")

32
dental_clinical_management/models/insurance_company.py

@ -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")

38
dental_clinical_management/models/medical_questionnaire.py

@ -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")

51
dental_clinical_management/models/medical_questions.py

@ -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()

33
dental_clinical_management/models/medicine_frequency.py

@ -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")

30
dental_clinical_management/models/treatment_category.py

@ -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")

64
dental_clinical_management/report/dental_appointment_card.xml

@ -0,0 +1,64 @@
<?xml version="1.0" encoding="UTF-8" ?>
<odoo>
<!-- Appointment card report template -->
<template id="appointment_card">
<t t-call="web.html_container">
<section>
<div style="width: 500px; height: 260px; border: 2px black solid; border-radius: 5px;">
<div class="container"
style="width: 500px; height: 50px; background-color:red">
<span>
<i class="fa fa-wave-pulse" style="color:black;"/>
</span>
<h1 style="margin-left: 150px; padding-top: 10px; color:white; border-radius: 10px;">
Appointment Card
</h1>
<div class="page">
<h1 style="margin-left: 15px;">
<b>Token :
<t t-esc="token"/>
</b>
<br/>
<label style="font-size: 18px;">
<b>Doctor :
<t t-esc="doctor"/>
</b>
</label>
</h1>
<p style="margin-left: 15px;">
<label>
<b>Specialised :</b>
<t t-esc="specialised"/>
</label>
<br/>
<label>
<b>Time :</b>
<t t-esc="appointment_time"/>
</label>
<br/>
<label>
<b>Date :</b>
<t t-esc="date"/>
</label>
<div style="position:absolute; left:320px; top:100px;">
<img t-attf-src="{{ request.httprequest.url_root }}dental_clinical_management/static/src/img/appointment.jpeg"
style="height: 140px; width: 150px;"/>
</div>
</p>
</div>
</div>
</div>
</section>
</t>
</template>
<!-- Action report appointment card -->
<record id="action_appointment_card" model="ir.actions.report">
<field name="name">Appointment Card</field>
<field name="model">dental.appointment</field>
<field name="report_type">qweb-pdf</field>
<field name="report_name">dental_clinical_management.appointment_card</field>
<field name="report_file">dental_clinical_management.appointment_card</field>
<field name="binding_model_id"
ref="dental_clinical_management.model_dental_appointment"/>
</record>
</odoo>

14
dental_clinical_management/report/dental_prescription_report.xml

@ -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_clinical_management.report_prescription</field>
<field name="report_file">dental_clinical_management.report_prescription</field>
<field name="print_report_name">'Prescription Report - %s' % (object.sequence_no)</field>
<field name="binding_model_id" ref="dental_clinical_management.model_dental_prescription"/>
<field name="binding_type">report</field>
</record>
</odoo>

119
dental_clinical_management/report/dental_prescription_templates.xml

@ -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>

23
dental_clinical_management/security/dental_clinical_management_groups.xml

@ -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>

33
dental_clinical_management/security/dental_clinical_management_security.xml

@ -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_clinical_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_clinical_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_clinical_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_clinical_management.group_dental_manager'))]"/>
</record>
</odoo>

27
dental_clinical_management/security/ir.model.access.csv

@ -0,0 +1,27 @@
id,name,model_id:id,group_id:id,perm_read,perm_write,perm_create,perm_unlink
access_dental_time_shift,access.dental.time.shift,model_dental_time_shift,base.group_user,1,1,1,1
access_medicine_frequency,access.medicine.frequency,model_medicine_frequency,base.group_user,1,1,1,1
access_medical_questions,access.medical.questions,model_medical_questions,base.group_user,1,1,1,1
access_medical_questionnaire,access.medical.questionnaire,model_medical_questionnaire,base.group_user,1,1,1,1
access_xray_report,access.xray.report,model_xray_report,base.group_user,1,1,1,1
access_portal_dental_prescription,access.portal.dental.prescription,model_dental_prescription,base.group_portal,1,0,0,0
access_portal_dental_prescription_lines,access.portal.dental.prescription_lines,model_dental_prescription_lines,base.group_portal,1,0,0,0
access_portal_dental_appointment,access.portal.dental.appointment,model_dental_appointment,base.group_portal,1,0,0,0
access_portal_dental_specialist,access.portal.dental.specialist,model_dental_specialist,base.group_portal,1,0,0,0
access_portal_dental_treatment,access.portal.dental.treatment,model_dental_treatment,base.group_portal,1,0,0,0
access_portal_product_template,access.portal.product.template,model_product_template,base.group_portal,1,0,0,0
access_portal_medicine_frequency,access.portal.medicine.frequency,model_medicine_frequency,base.group_portal,1,0,0,0
access_user_dental_appointment,access.user.dental.appointment,model_dental_appointment,dental_clinical_management.group_dental_doctor,1,1,0,0
access_user_dental_prescription,access.user.dental.prescription,model_dental_prescription,dental_clinical_management.group_dental_doctor,1,1,1,1
access_user_dental_prescription_lines,access.user.dental.prescription_lines,model_dental_prescription_lines,dental_clinical_management.group_dental_doctor,1,1,1,1
access_user_dental_specialist,access.user.dental.specialist,model_dental_specialist,dental_clinical_management.group_dental_doctor,1,0,0,0
access_user_treatment_category,access.user.treatment.category,model_treatment_category,dental_clinical_management.group_dental_doctor,1,0,0,0
access_user_dental_treatment,access.user.dental.treatment,model_dental_treatment,dental_clinical_management.group_dental_doctor,1,0,0,0
access_user_insurance_company,access.user.insurance.company,model_insurance_company,dental_clinical_management.group_dental_doctor,1,0,0,0
access_manager_dental_appointment,access.manager.dental.appointment,model_dental_appointment,dental_clinical_management.group_dental_manager,1,1,1,1
access_manager_dental_prescription,access.manager.dental.prescription,model_dental_prescription,dental_clinical_management.group_dental_manager,1,1,1,1
access_manager_dental_prescription_lines,access.manager.dental.prescription_lines,model_dental_prescription_lines,dental_clinical_management.group_dental_manager,1,1,1,1
access_manager_dental_specialist,access.manager.dental.specialist,model_dental_specialist,dental_clinical_management.group_dental_manager,1,1,1,1
access_manager_treatment_category,access.manager.treatment.category,model_treatment_category,dental_clinical_management.group_dental_manager,1,1,1,1
access_manager_dental_treatment,access.manager.dental.treatment,model_dental_treatment,dental_clinical_management.group_dental_manager,1,1,1,1
access_manager_insurance_company,access.manager.insurance.company,model_insurance_company,dental_clinical_management.group_dental_manager,1,1,1,1
1 id name model_id:id group_id:id perm_read perm_write perm_create perm_unlink
2 access_dental_time_shift access.dental.time.shift model_dental_time_shift base.group_user 1 1 1 1
3 access_medicine_frequency access.medicine.frequency model_medicine_frequency base.group_user 1 1 1 1
4 access_medical_questions access.medical.questions model_medical_questions base.group_user 1 1 1 1
5 access_medical_questionnaire access.medical.questionnaire model_medical_questionnaire base.group_user 1 1 1 1
6 access_xray_report access.xray.report model_xray_report base.group_user 1 1 1 1
7 access_portal_dental_prescription access.portal.dental.prescription model_dental_prescription base.group_portal 1 0 0 0
8 access_portal_dental_prescription_lines access.portal.dental.prescription_lines model_dental_prescription_lines base.group_portal 1 0 0 0
9 access_portal_dental_appointment access.portal.dental.appointment model_dental_appointment base.group_portal 1 0 0 0
10 access_portal_dental_specialist access.portal.dental.specialist model_dental_specialist base.group_portal 1 0 0 0
11 access_portal_dental_treatment access.portal.dental.treatment model_dental_treatment base.group_portal 1 0 0 0
12 access_portal_product_template access.portal.product.template model_product_template base.group_portal 1 0 0 0
13 access_portal_medicine_frequency access.portal.medicine.frequency model_medicine_frequency base.group_portal 1 0 0 0
14 access_user_dental_appointment access.user.dental.appointment model_dental_appointment dental_clinical_management.group_dental_doctor 1 1 0 0
15 access_user_dental_prescription access.user.dental.prescription model_dental_prescription dental_clinical_management.group_dental_doctor 1 1 1 1
16 access_user_dental_prescription_lines access.user.dental.prescription_lines model_dental_prescription_lines dental_clinical_management.group_dental_doctor 1 1 1 1
17 access_user_dental_specialist access.user.dental.specialist model_dental_specialist dental_clinical_management.group_dental_doctor 1 0 0 0
18 access_user_treatment_category access.user.treatment.category model_treatment_category dental_clinical_management.group_dental_doctor 1 0 0 0
19 access_user_dental_treatment access.user.dental.treatment model_dental_treatment dental_clinical_management.group_dental_doctor 1 0 0 0
20 access_user_insurance_company access.user.insurance.company model_insurance_company dental_clinical_management.group_dental_doctor 1 0 0 0
21 access_manager_dental_appointment access.manager.dental.appointment model_dental_appointment dental_clinical_management.group_dental_manager 1 1 1 1
22 access_manager_dental_prescription access.manager.dental.prescription model_dental_prescription dental_clinical_management.group_dental_manager 1 1 1 1
23 access_manager_dental_prescription_lines access.manager.dental.prescription_lines model_dental_prescription_lines dental_clinical_management.group_dental_manager 1 1 1 1
24 access_manager_dental_specialist access.manager.dental.specialist model_dental_specialist dental_clinical_management.group_dental_manager 1 1 1 1
25 access_manager_treatment_category access.manager.treatment.category model_treatment_category dental_clinical_management.group_dental_manager 1 1 1 1
26 access_manager_dental_treatment access.manager.dental.treatment model_dental_treatment dental_clinical_management.group_dental_manager 1 1 1 1
27 access_manager_insurance_company access.manager.insurance.company model_insurance_company dental_clinical_management.group_dental_manager 1 1 1 1

BIN
dental_clinical_management/static/description/assets/icons/capture (1).png

Binary file not shown.

After

Width:  |  Height:  |  Size: 36 KiB

BIN
dental_clinical_management/static/description/assets/icons/check.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 3.6 KiB

BIN
dental_clinical_management/static/description/assets/icons/chevron.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 310 B

BIN
dental_clinical_management/static/description/assets/icons/cogs.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.3 KiB

BIN
dental_clinical_management/static/description/assets/icons/consultation.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.4 KiB

BIN
dental_clinical_management/static/description/assets/icons/ecom-black.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 576 B

BIN
dental_clinical_management/static/description/assets/icons/education-black.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 733 B

BIN
dental_clinical_management/static/description/assets/icons/hotel-black.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 911 B

BIN
dental_clinical_management/static/description/assets/icons/img.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.1 KiB

BIN
dental_clinical_management/static/description/assets/icons/license.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.1 KiB

BIN
dental_clinical_management/static/description/assets/icons/lifebuoy.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.2 KiB

BIN
dental_clinical_management/static/description/assets/icons/manufacturing-black.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 673 B

BIN
dental_clinical_management/static/description/assets/icons/photo-capture.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 11 KiB

BIN
dental_clinical_management/static/description/assets/icons/pos-black.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 878 B

BIN
dental_clinical_management/static/description/assets/icons/puzzle.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 653 B

BIN
dental_clinical_management/static/description/assets/icons/restaurant-black.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 905 B

BIN
dental_clinical_management/static/description/assets/icons/service-black.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 839 B

BIN
dental_clinical_management/static/description/assets/icons/trading-black.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 427 B

BIN
dental_clinical_management/static/description/assets/icons/training.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 627 B

BIN
dental_clinical_management/static/description/assets/icons/update.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.2 KiB

BIN
dental_clinical_management/static/description/assets/icons/user.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 988 B

BIN
dental_clinical_management/static/description/assets/icons/wrench.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.2 KiB

BIN
dental_clinical_management/static/description/assets/misc/Cybrosys R.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 80 KiB

33
dental_clinical_management/static/description/assets/misc/email.svg

@ -0,0 +1,33 @@
<svg width="80" height="81" viewBox="0 0 80 81" fill="none" xmlns="http://www.w3.org/2000/svg">
<g id="3116889_design_email_material_communication_mail_icon 1" clip-path="url(#clip0_81_366)">
<g id="layer1">
<path id="rect3851" d="M74.6067 0.730957H5.5424C2.75742 0.730957 0.499756 3.01685 0.499756 5.83664V75.7642C0.499756 78.584 2.75742 80.8699 5.5424 80.8699H74.6067C77.3916 80.8699 79.6493 78.584 79.6493 75.7642V5.83664C79.6493 3.01685 77.3916 0.730957 74.6067 0.730957Z" fill="#DB534B"/>
<g id="Clip path group">
<mask id="mask0_81_366" style="mask-type:luminance" maskUnits="userSpaceOnUse" x="1" y="5" width="78" height="76">
<g id="clipPath4206">
<path id="rect4208" d="M73.6244 5.2915H6.62595C3.92428 5.2915 1.73413 7.4473 1.73413 10.1066V76.0546C1.73413 78.7139 3.92428 80.8697 6.62595 80.8697H73.6244C76.3261 80.8697 78.5162 78.7139 78.5162 76.0546V10.1066C78.5162 7.4473 76.3261 5.2915 73.6244 5.2915Z" fill="white"/>
</g>
</mask>
<g mask="url(#mask0_81_366)">
<g id="g4145" opacity="0.489612">
<g id="g4147">
<path id="path4149" d="M65.8115 41.5171C65.8115 54.9863 54.4292 65.9053 40.3884 65.9053L198.828 221.861C212.869 221.861 224.251 210.942 224.251 197.472L65.8115 41.5171Z" fill="black" fill-opacity="0.0588235"/>
<path id="path4151" d="M40.3884 65.9051C33.2495 65.9051 26.7979 63.0825 22.1802 58.5371L180.62 214.492C185.237 219.038 191.689 221.86 198.828 221.86L40.3884 65.9051Z" fill="black" fill-opacity="0.0588235"/>
<path id="path4153" d="M22.1802 58.5373C17.7157 54.1428 14.9653 48.1381 14.9653 41.5171L173.405 197.472C173.405 204.093 176.155 210.098 180.62 214.493L22.1802 58.5373Z" fill="black" fill-opacity="0.0588235"/>
<path id="path4155" d="M14.9653 41.5171C14.9653 28.0479 26.3476 17.1289 40.3884 17.1289L198.828 173.084C184.787 173.084 173.405 184.003 173.405 197.472L14.9653 41.5171Z" fill="black" fill-opacity="0.0588235"/>
<path id="path4157" d="M40.3884 17.1289C47.5273 17.1289 53.9789 19.9516 58.5966 24.4969L217.036 180.452C212.418 175.907 205.967 173.084 198.828 173.084L40.3884 17.1289Z" fill="black" fill-opacity="0.0588235"/>
<path id="path4159" d="M58.5964 24.4971C63.0609 28.8916 65.8113 34.8963 65.8113 41.5173L224.251 197.473C224.251 190.852 221.5 184.847 217.036 180.452L58.5964 24.4971Z" fill="black" fill-opacity="0.0588235"/>
</g>
<path id="path4111" d="M65.8114 41.5171C65.8114 54.9863 54.4291 65.9053 40.3884 65.9053C26.3476 65.9053 14.9653 54.9863 14.9653 41.5171C14.9653 28.0479 26.3476 17.1289 40.3884 17.1289C54.4291 17.1289 65.8114 28.0479 65.8114 41.5171Z" fill="black" fill-opacity="0.0588235"/>
</g>
</g>
</g>
<path id="path3864" d="M17.506 17.5386H62.9018C64.4068 17.5386 65.8501 18.1439 66.9143 19.2214C67.9784 20.2988 68.5763 21.7602 68.5763 23.284V57.7564C68.5763 58.5109 68.4295 59.258 68.1443 59.9551C67.8592 60.6521 67.4412 61.2855 66.9143 61.819C66.3873 62.3525 65.7618 62.7757 65.0733 63.0645C64.3849 63.3532 63.647 63.5018 62.9018 63.5018H17.506C14.3567 63.5018 11.8315 60.9164 11.8315 57.7564V23.284C11.8315 20.0953 14.3567 17.5386 17.506 17.5386ZM40.2039 37.6475L62.9018 23.284H17.506L40.2039 37.6475ZM17.506 57.7564H62.9018V30.0923L40.2039 44.4271L17.506 30.0923V57.7564Z" fill="white"/>
</g>
</g>
<defs>
<clipPath id="clip0_81_366">
<rect width="80" height="81" fill="white"/>
</clipPath>
</defs>
</svg>

After

Width:  |  Height:  |  Size: 3.2 KiB

3
dental_clinical_management/static/description/assets/misc/phone.svg

@ -0,0 +1,3 @@
<svg width="36" height="44" viewBox="0 0 36 44" fill="none" xmlns="http://www.w3.org/2000/svg">
<path id="Vector" d="M7.25 19.3903C10.13 26.0689 14.76 31.5322 20.43 34.9305L24.83 29.7268C25.38 29.0778 26.17 28.889 26.86 29.1486C29.1 30.0218 31.51 30.4938 34 30.4938C35.11 30.4938 36 31.544 36 32.8537V41.1135C36 42.4233 35.11 43.4734 34 43.4734C15.22 43.4734 0 25.5143 0 3.35456C0 2.0448 0.9 0.994629 2 0.994629H9C10.11 0.994629 11 2.0448 11 3.35456C11 6.29268 11.4 9.1364 12.14 11.7795C12.36 12.5937 12.2 13.5259 11.65 14.1749L7.25 19.3903Z" fill="white"/>
</svg>

After

Width:  |  Height:  |  Size: 565 B

9
dental_clinical_management/static/description/assets/misc/star (1) 2.svg

File diff suppressed because one or more lines are too long

After

Width:  |  Height:  |  Size: 26 KiB

9
dental_clinical_management/static/description/assets/misc/support (1) 1.svg

File diff suppressed because one or more lines are too long

After

Width:  |  Height:  |  Size: 43 KiB

6
dental_clinical_management/static/description/assets/misc/support-email.svg

@ -0,0 +1,6 @@
<svg width="49" height="37" viewBox="0 0 49 37" fill="none" xmlns="http://www.w3.org/2000/svg">
<g id="Group">
<path id="Vector" d="M2.23798 3.59132C3.53363 4.39742 21.5313 15.9748 22.2027 16.3917C22.8741 16.8087 23.5573 17.0032 24.6173 17.0032C25.6774 17.0032 26.3606 16.8087 27.0319 16.3917C27.7033 15.9748 45.701 4.39742 46.9967 3.59132C47.4796 3.29945 48.2923 2.77131 48.469 2.17368C48.7753 1.11741 48.4455 0.714355 47.138 0.714355H24.6173H2.09664C0.789214 0.714355 0.459412 1.13131 0.765656 2.17368C0.942335 2.78521 1.75506 3.29945 2.23798 3.59132Z" fill="white"/>
<path id="Vector_2" d="M48.0214 4.21664C47.0555 4.80037 38.3865 12.0831 32.6503 16.4611L42.3323 29.3171C42.5679 29.5951 42.6739 29.9286 42.5443 30.0954C42.403 30.2483 42.0967 30.1649 41.8494 29.9008L30.2357 18.3374C28.4807 19.6716 27.2439 20.5889 27.0319 20.7279C26.1249 21.2699 25.4889 21.3394 24.6173 21.3394C23.7457 21.3394 23.1096 21.2699 22.2027 20.7279C21.9789 20.5889 20.7539 19.6716 18.9989 18.3374L7.38519 29.9008C7.14961 30.1788 6.83159 30.2622 6.69025 30.0954C6.54891 29.9425 6.65491 29.5951 6.89048 29.3171L16.5607 16.4611C10.8245 12.0831 2.06126 4.80037 1.09541 4.21664C0.0588929 3.59121 0 4.32783 0 4.89766C0 5.46749 0 33.3893 0 33.3893C0 34.6819 1.61367 36.2941 2.76797 36.2941H24.6173H46.4666C47.6209 36.2941 48.999 34.668 48.999 33.3893C48.999 33.3893 48.999 5.4536 48.999 4.89766C48.999 4.31393 49.0697 3.59121 48.0214 4.21664Z" fill="white"/>
</g>
</svg>

After

Width:  |  Height:  |  Size: 1.4 KiB

17
dental_clinical_management/static/description/assets/misc/tick-mark.svg

@ -0,0 +1,17 @@
<svg width="52" height="52" viewBox="0 0 52 52" fill="none" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink">
<rect width="52" height="52" fill="#F5F5F5"/>
<g clip-path="url(#clip0_0_1)">
<rect width="1440" height="7504" transform="translate(-107 -1660)" fill="white"/>
<rect x="-45" y="-203" width="1305" height="937" rx="19" fill="#FFF5FC"/>
<rect width="52" height="52" fill="url(#pattern0)"/>
</g>
<defs>
<pattern id="pattern0" patternContentUnits="objectBoundingBox" width="1" height="1">
<use xlink:href="#image0_0_1" transform="scale(0.00387597)"/>
</pattern>
<clipPath id="clip0_0_1">
<rect width="1440" height="7504" fill="white" transform="translate(-107 -1660)"/>
</clipPath>
<image id="image0_0_1" width="258" height="258" xlink:href=""/>
</defs>
</svg>

After

Width:  |  Height:  |  Size: 4.0 KiB

9
dental_clinical_management/static/description/assets/misc/whatsapp 1.svg

File diff suppressed because one or more lines are too long

After

Width:  |  Height:  |  Size: 38 KiB

33
dental_clinical_management/static/description/assets/misc/whatsapp.svg

@ -0,0 +1,33 @@
<svg width="80" height="80" viewBox="0 0 80 80" fill="none" xmlns="http://www.w3.org/2000/svg">
<g id="3116884_whatsapp_square_chat_design_message_icon 1" clip-path="url(#clip0_81_382)">
<g id="layer1">
<path id="rect3851" d="M74.6066 0.72168H5.5424C2.75742 0.72168 0.499756 2.97935 0.499756 5.76433V74.8286C0.499756 77.6135 2.75742 79.8712 5.5424 79.8712H74.6066C77.3916 79.8712 79.6492 77.6135 79.6492 74.8286V5.76433C79.6492 2.97935 77.3916 0.72168 74.6066 0.72168Z" fill="#39BB59"/>
<g id="Clip path group">
<mask id="mask0_81_382" style="mask-type:luminance" maskUnits="userSpaceOnUse" x="6" y="9" width="75" height="72">
<g id="clipPath4206">
<path id="rect4208" d="M75.7716 9.01709H11.1629C8.55758 9.01709 6.44556 11.0471 6.44556 13.5512V75.6502C6.44556 78.1543 8.55758 80.1843 11.1629 80.1843H75.7716C78.3769 80.1843 80.4889 78.1543 80.4889 75.6502V13.5512C80.4889 11.0471 78.3769 9.01709 75.7716 9.01709Z" fill="white"/>
</g>
</mask>
<g mask="url(#mask0_81_382)">
<g id="g4145" opacity="0.489612">
<g id="g4147">
<path id="path4149" d="M68.2374 43.1284C68.2374 55.8115 57.2611 66.0932 43.7212 66.0932L196.51 212.946C210.049 212.946 221.026 202.665 221.026 189.982L68.2374 43.1284Z" fill="black" fill-opacity="0.0588235"/>
<path id="path4151" d="M43.7211 66.0932C36.8369 66.0932 30.6154 63.4353 26.1624 59.1553L178.951 206.008C183.404 210.289 189.625 212.946 196.51 212.946L43.7211 66.0932Z" fill="black" fill-opacity="0.0588235"/>
<path id="path4153" d="M26.1623 59.1553C21.8571 55.0173 19.2048 49.363 19.2048 43.1284L171.993 189.982C171.993 196.216 174.645 201.87 178.951 206.008L26.1623 59.1553Z" fill="black" fill-opacity="0.0588235"/>
<path id="path4155" d="M19.2048 43.1284C19.2048 30.4453 30.1811 20.1636 43.7211 20.1636L196.509 167.017C182.969 167.017 171.993 177.299 171.993 189.982L19.2048 43.1284Z" fill="black" fill-opacity="0.0588235"/>
<path id="path4157" d="M43.7212 20.1636C50.6054 20.1636 56.8269 22.8215 61.2799 27.1015L214.068 173.955C209.615 169.675 203.394 167.017 196.51 167.017L43.7212 20.1636Z" fill="black" fill-opacity="0.0588235"/>
<path id="path4159" d="M61.2798 27.1016C65.585 31.2396 68.2373 36.8939 68.2373 43.1284L221.026 189.982C221.026 183.747 218.373 178.093 214.068 173.955L61.2798 27.1016Z" fill="black" fill-opacity="0.0588235"/>
</g>
<path id="path4111" d="M68.2373 43.1284C68.2373 55.8115 57.261 66.0932 43.7211 66.0932C30.1811 66.0932 19.2048 55.8115 19.2048 43.1284C19.2048 30.4453 30.1811 20.1636 43.7211 20.1636C57.261 20.1636 68.2373 30.4453 68.2373 43.1284Z" fill="black" fill-opacity="0.0588235"/>
</g>
</g>
</g>
<path id="path4074" d="M51.3896 43.6875C51.9673 43.9879 52.337 44.1497 52.4526 44.3808C52.5912 44.635 52.545 45.7904 51.9673 47.1076C51.5051 48.4017 49.1018 49.6496 48.0388 49.6958C46.9758 49.7421 46.9527 50.5277 41.1985 48.0089C35.4444 45.49 31.9781 39.3431 31.7008 38.9502C31.4235 38.5574 29.4823 35.7612 29.5748 32.9188C29.6903 30.0995 31.1693 28.7592 31.7701 28.2046C32.3247 27.6037 32.9487 27.5344 33.3415 27.6037H34.4276C34.7743 27.6037 35.2596 27.4651 35.6986 28.6437L37.2931 32.965C37.4318 33.2654 37.5242 33.6121 37.3163 33.9818L36.6923 34.9293L35.7911 35.8998C35.5138 36.1771 35.1902 36.4776 35.5138 37.0553C35.7911 37.6561 36.9465 39.5741 38.5641 41.1687C40.667 43.2022 42.5158 43.8724 43.0704 44.1728C43.625 44.4963 43.9716 44.4501 44.3182 44.0804L46.1901 41.9081C46.6291 41.3304 46.9989 41.4691 47.5304 41.6539L51.3896 43.6875ZM40.4128 16.0493C46.5417 16.0493 52.4195 18.484 56.7533 22.8178C61.0871 27.1515 63.5217 33.0293 63.5217 39.1582C63.5217 45.287 61.0871 51.1649 56.7533 55.4986C52.4195 59.8324 46.5417 62.2671 40.4128 62.2671C35.8604 62.2671 31.6315 60.9498 28.0496 58.6852L17.304 62.2671L20.8858 51.5214C18.6212 47.9396 17.304 43.7106 17.304 39.1582C17.304 33.0293 19.7386 27.1515 24.0724 22.8178C28.4061 18.484 34.284 16.0493 40.4128 16.0493ZM40.4128 20.6711C35.5098 20.6711 30.8075 22.6188 27.3405 26.0858C23.8735 29.5528 21.9257 34.2551 21.9257 39.1582C21.9257 43.1329 23.1736 46.8072 25.2996 49.8114L23.0812 56.4898L29.7596 54.2714C32.7638 56.3974 36.4381 57.6453 40.4128 57.6453C45.3159 57.6453 50.0182 55.6975 53.4852 52.2305C56.9522 48.7635 58.9 44.0613 58.9 39.1582C58.9 34.2551 56.9522 29.5528 53.4852 26.0858C50.0182 22.6188 45.3159 20.6711 40.4128 20.6711Z" fill="white"/>
</g>
</g>
<defs>
<clipPath id="clip0_81_382">
<rect width="80" height="80" fill="white"/>
</clipPath>
</defs>
</svg>

After

Width:  |  Height:  |  Size: 4.3 KiB

BIN
dental_clinical_management/static/description/assets/modules/1.jpg

Binary file not shown.

After

Width:  |  Height:  |  Size: 87 KiB

BIN
dental_clinical_management/static/description/assets/modules/2.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 80 KiB

BIN
dental_clinical_management/static/description/assets/modules/3.jpg

Binary file not shown.

After

Width:  |  Height:  |  Size: 95 KiB

BIN
dental_clinical_management/static/description/assets/modules/4.jpg

Binary file not shown.

After

Width:  |  Height:  |  Size: 78 KiB

BIN
dental_clinical_management/static/description/assets/modules/5.jpg

Binary file not shown.

After

Width:  |  Height:  |  Size: 164 KiB

BIN
dental_clinical_management/static/description/assets/modules/6.jpg

Binary file not shown.

After

Width:  |  Height:  |  Size: 81 KiB

BIN
dental_clinical_management/static/description/assets/screenshots/10.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 80 KiB

BIN
dental_clinical_management/static/description/assets/screenshots/11.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 60 KiB

BIN
dental_clinical_management/static/description/assets/screenshots/12.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 36 KiB

BIN
dental_clinical_management/static/description/assets/screenshots/13.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 97 KiB

BIN
dental_clinical_management/static/description/assets/screenshots/14.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 122 KiB

BIN
dental_clinical_management/static/description/assets/screenshots/15.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 96 KiB

BIN
dental_clinical_management/static/description/assets/screenshots/16.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 86 KiB

BIN
dental_clinical_management/static/description/assets/screenshots/17.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 102 KiB

BIN
dental_clinical_management/static/description/assets/screenshots/18.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 102 KiB

BIN
dental_clinical_management/static/description/assets/screenshots/19.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 61 KiB

BIN
dental_clinical_management/static/description/assets/screenshots/2.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 38 KiB

BIN
dental_clinical_management/static/description/assets/screenshots/20.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 52 KiB

BIN
dental_clinical_management/static/description/assets/screenshots/21.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 90 KiB

BIN
dental_clinical_management/static/description/assets/screenshots/22.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 70 KiB

BIN
dental_clinical_management/static/description/assets/screenshots/23.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 71 KiB

BIN
dental_clinical_management/static/description/assets/screenshots/24.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 48 KiB

BIN
dental_clinical_management/static/description/assets/screenshots/25.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 46 KiB

BIN
dental_clinical_management/static/description/assets/screenshots/26.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 115 KiB

BIN
dental_clinical_management/static/description/assets/screenshots/27.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 40 KiB

BIN
dental_clinical_management/static/description/assets/screenshots/28.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 48 KiB

BIN
dental_clinical_management/static/description/assets/screenshots/29.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 80 KiB

BIN
dental_clinical_management/static/description/assets/screenshots/3.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 32 KiB

BIN
dental_clinical_management/static/description/assets/screenshots/30.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 101 KiB

BIN
dental_clinical_management/static/description/assets/screenshots/31.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 50 KiB

BIN
dental_clinical_management/static/description/assets/screenshots/32.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 41 KiB

BIN
dental_clinical_management/static/description/assets/screenshots/33.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 132 KiB

BIN
dental_clinical_management/static/description/assets/screenshots/34.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 93 KiB

Some files were not shown because too many files changed in this diff

Loading…
Cancel
Save