You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					
					
						
							302 lines
						
					
					
						
							15 KiB
						
					
					
				
			
		
		
		
			
			
			
		
		
	
	
							302 lines
						
					
					
						
							15 KiB
						
					
					
				
								# -*- 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 xlrd.xlsx import ET
							 | 
						|
								from odoo import api, fields, models, _
							 | 
						|
								from odoo.exceptions import ValidationError
							 | 
						|
								
							 | 
						|
								
							 | 
						|
								class DynamicFields(models.Model):
							 | 
						|
								    """model to create dynamic fields to selected model"""
							 | 
						|
								    _name = 'dynamic.fields'
							 | 
						|
								    _rec_name = 'field_description'
							 | 
						|
								    _description = 'Custom Dynamic Fields'
							 | 
						|
								    _inherit = 'ir.model.fields'
							 | 
						|
								
							 | 
						|
								    @api.model
							 | 
						|
								    def get_possible_field_types(self):
							 | 
						|
								        """Return all available field types other than 'one2many' and
							 | 
						|
								        'reference' fields."""
							 | 
						|
								        field_list = sorted((key, key) for key in fields.MetaField.by_type)
							 | 
						|
								        field_list.remove(('one2many', 'one2many'))
							 | 
						|
								        field_list.remove(('reference', 'reference'))
							 | 
						|
								        return field_list
							 | 
						|
								
							 | 
						|
								    position_field_id = fields.Many2one('ir.model.fields',
							 | 
						|
								                                        string='Field Name',
							 | 
						|
								                                        help='Field name',
							 | 
						|
								                                        required=True,
							 | 
						|
								                                        ondelete='cascade')
							 | 
						|
								    position = fields.Selection([('before', 'Before'),
							 | 
						|
								                                 ('after', 'After')], string='Position',
							 | 
						|
								                                help='Position of new field to selected field',
							 | 
						|
								                                required=True)
							 | 
						|
								    model_id = fields.Many2one('ir.model', string='Model',
							 | 
						|
								                               required=True,
							 | 
						|
								                               index=True, ondelete='cascade',
							 | 
						|
								                               help="The model this field belongs to")
							 | 
						|
								    ref_model_id = fields.Many2one('ir.model', string='Model',
							 | 
						|
								                                   index=True, help='Reference model')
							 | 
						|
								    selection_field = fields.Char(string="Selection Options",
							 | 
						|
								                                  help='Selection values given to the newly '
							 | 
						|
								                                       'created field')
							 | 
						|
								    rel_field_id = fields.Many2one('ir.model.fields',
							 | 
						|
								                                   string='Related Field', help='Related field')
							 | 
						|
								    field_type = fields.Selection(selection='get_possible_field_types',
							 | 
						|
								                                  string='Field Type',
							 | 
						|
								                                  required=True, help='Field type of the '
							 | 
						|
								                                                      'newly created field')
							 | 
						|
								    ttype = fields.Selection(string="Field Type", related='field_type',
							 | 
						|
								                             help='field type')
							 | 
						|
								    widget_id = fields.Many2one('dynamic.field.widgets',
							 | 
						|
								                                string='Widget',
							 | 
						|
								                                help='widget given to the newly '
							 | 
						|
								                                     'creating field')
							 | 
						|
								    groups = fields.Many2many('res.groups',
							 | 
						|
								                              'employee_dynamic_fields_group_rel',
							 | 
						|
								                              'field_id', 'group_id',
							 | 
						|
								                              string='Groups', help="Groups")
							 | 
						|
								    is_extra_features = fields.Boolean(string="Show Extra Properties",
							 | 
						|
								                                       help='Enable to give extra features to '
							 | 
						|
								                                            'the newly creating field')
							 | 
						|
								    status = fields.Selection([
							 | 
						|
								        ('draft', 'Draft'),
							 | 
						|
								        ('form', 'Field Created'),
							 | 
						|
								        ('tree', 'Added in Tree View'),
							 | 
						|
								    ], string='Status', index=True, readonly=True, tracking=True,
							 | 
						|
								        copy=False, default='draft',
							 | 
						|
								        help='Record Status')
							 | 
						|
								
							 | 
						|
								    form_view_id = fields.Many2one('ir.ui.view',
							 | 
						|
								                                   string="Form View ID",
							 | 
						|
								                                   help='Form view id of selected model',
							 | 
						|
								                                   required=True)
							 | 
						|
								    form_view_inherit_id = fields.Char(string="Form View Inherit Id",
							 | 
						|
								                                       help='External id of selected form view',
							 | 
						|
								                                       related='form_view_id.xml_id')
							 | 
						|
								    is_add_field_in_tree = fields.Boolean(string="Add Field to the Tree View",
							 | 
						|
								                                          help='Enable to add field in tree '
							 | 
						|
								                                               'view')
							 | 
						|
								    tree_view_id = fields.Many2one('ir.ui.view',
							 | 
						|
								                                   string="Tree View ID",
							 | 
						|
								                                   help='select a tree view id')
							 | 
						|
								    tree_view_inherit_id = fields.Char(string="Tree View Inherit Id",
							 | 
						|
								                                       help='External id of selected tree view',
							 | 
						|
								                                       related='tree_view_id.xml_id')
							 | 
						|
								    created_form_view_id = fields.Many2one('ir.ui.view',
							 | 
						|
								                                           help='Created from view id',
							 | 
						|
								                                           string='Created Form View')
							 | 
						|
								    created_tree_view_id = fields.Many2one('ir.ui.view',
							 | 
						|
								                                           help='Created tree view id',
							 | 
						|
								                                           string='Created Tree View')
							 | 
						|
								    tree_field_id = fields.Many2one('ir.model.fields',
							 | 
						|
								                                    sting='Tree Field',
							 | 
						|
								                                    domain="[('id', 'in', "
							 | 
						|
								                                           "tree_field_ids)]",
							 | 
						|
								                                    help='Tree view fields of selected tree '
							 | 
						|
								                                         'view')
							 | 
						|
								    tree_field_position = fields.Selection([('before', 'Before'),
							 | 
						|
								                                            ('after', 'After')],
							 | 
						|
								                                           string='Tree Field Position',
							 | 
						|
								                                           help='Position of new field to '
							 | 
						|
								                                                'selected field in tree view')
							 | 
						|
								    is_field_in_tree = fields.Boolean(string='Field Visible In Tree View',
							 | 
						|
								                                      help='Enable this to enable the toggle '
							 | 
						|
								                                           'view of this field in tree view')
							 | 
						|
								    tree_field_ids = fields.Many2many('ir.model.fields',
							 | 
						|
								                                      string='Tree View Fields',
							 | 
						|
								                                      compute='_compute_tree_field_ids',
							 | 
						|
								                                      help='Domain field for tree view fields')
							 | 
						|
								
							 | 
						|
								    @api.depends('tree_view_id')
							 | 
						|
								    def _compute_tree_field_ids(self):
							 | 
						|
								        """Compute function to find the tree view fields of selected tree view
							 | 
						|
								        in field tree_view_id"""
							 | 
						|
								        for rec in self:
							 | 
						|
								            if rec.tree_view_id:
							 | 
						|
								                field_list = []
							 | 
						|
								                if rec.tree_view_id.xml_id:
							 | 
						|
								                    tree_fields = ET.fromstring(self.env.ref(
							 | 
						|
								                        rec.tree_view_id.xml_id).arch).findall(".//field")
							 | 
						|
								                    for field in tree_fields:
							 | 
						|
								                        field_list.append(field.get('name'))
							 | 
						|
								                inherit_id = rec.tree_view_id.inherit_id \
							 | 
						|
								                    if rec.tree_view_id.inherit_id else False
							 | 
						|
								                while inherit_id:
							 | 
						|
								                    if inherit_id.xml_id:
							 | 
						|
								                        tree_fields = ET.fromstring(self.env.ref(
							 | 
						|
								                            inherit_id.xml_id).arch).findall(".//field")
							 | 
						|
								                        for field in tree_fields:
							 | 
						|
								                            field_list.append(field.get('name'))
							 | 
						|
								                    inherit_id = inherit_id.inherit_id \
							 | 
						|
								                        if inherit_id.inherit_id else False
							 | 
						|
								                self.tree_field_ids = self.env['ir.model.fields'].search(
							 | 
						|
								                    [('model_id', '=', self.model_id.id),
							 | 
						|
								                     ('name', 'in', field_list)])
							 | 
						|
								            else:
							 | 
						|
								                rec.tree_field_ids = False
							 | 
						|
								
							 | 
						|
								    def action_create_dynamic_fields(self):
							 | 
						|
								        """Function creates field with given properties to the selected model"""
							 | 
						|
								        self.write({'status': 'form'})
							 | 
						|
								        if self.field_type == 'monetary' and not self.env[
							 | 
						|
								            'ir.model.fields'].sudo().search([('model', '=', self.model_id.id),
							 | 
						|
								                                              ('name', '=', 'currency_id')]):
							 | 
						|
								            self.env['ir.model.fields'].sudo().create({
							 | 
						|
								                'name': 'x_currency_id',
							 | 
						|
								                'field_description': 'Currency',
							 | 
						|
								                'model_id': self.model_id.id,
							 | 
						|
								                'ttype': 'many2one',
							 | 
						|
								                'relation': 'res.currency',
							 | 
						|
								                'is_dynamic_field': True
							 | 
						|
								            })
							 | 
						|
								        self.env['ir.model.fields'].sudo().create({
							 | 
						|
								            'name': self.name,
							 | 
						|
								            'field_description': self.field_description,
							 | 
						|
								            'model_id': self.model_id.id,
							 | 
						|
								            'ttype': self.field_type,
							 | 
						|
								            'relation': self.ref_model_id.model,
							 | 
						|
								            'required': self.required,
							 | 
						|
								            'index': self.index,
							 | 
						|
								            'store': self.store,
							 | 
						|
								            'help': self.help,
							 | 
						|
								            'readonly': self.readonly,
							 | 
						|
								            'selection': self.selection_field,
							 | 
						|
								            'copied': self.copied,
							 | 
						|
								            'is_dynamic_field': True
							 | 
						|
								        })
							 | 
						|
								        inherit_form_view_name = str(
							 | 
						|
								            self.form_view_id.name) + ".inherit.dynamic.custom." + \
							 | 
						|
								                                 str(self.field_description) + ".field"
							 | 
						|
								        xml_id = self.form_view_id.xml_id
							 | 
						|
								        inherit_id = self.env.ref(xml_id)
							 | 
						|
								        arch_base = _('<?xml version="1.0"?>'
							 | 
						|
								                      '<data>'
							 | 
						|
								                      '<field name="%s" position="%s">'
							 | 
						|
								                      '<field name="%s"/>'
							 | 
						|
								                      '</field>'
							 | 
						|
								                      '</data>') % (self.position_field_id.name,
							 | 
						|
								                                    self.position, self.name)
							 | 
						|
								        if self.widget_id:
							 | 
						|
								            arch_base = _('<?xml version="1.0"?>'
							 | 
						|
								                          '<data>'
							 | 
						|
								                          '<field name="%s" position="%s">'
							 | 
						|
								                          '<field name="%s" widget="%s"/>'
							 | 
						|
								                          '</field>'
							 | 
						|
								                          '</data>') % (self.position_field_id.name,
							 | 
						|
								                                        self.position, self.name,
							 | 
						|
								                                        self.widget_id.name)
							 | 
						|
								        self.created_form_view_id = self.env['ir.ui.view'].sudo().create({
							 | 
						|
								            'name': inherit_form_view_name,
							 | 
						|
								            'type': 'form',
							 | 
						|
								            'model': self.model_id.model,
							 | 
						|
								            'mode': 'extension',
							 | 
						|
								            'inherit_id': inherit_id.id,
							 | 
						|
								            'arch_base': arch_base,
							 | 
						|
								            'active': True
							 | 
						|
								        })
							 | 
						|
								        return {
							 | 
						|
								            'type': 'ir.actions.client',
							 | 
						|
								            'tag': 'reload',
							 | 
						|
								        }
							 | 
						|
								
							 | 
						|
								    def action_add_field_to_tree_view(self):
							 | 
						|
								        """Function adds created field to the tree view"""
							 | 
						|
								        if self.is_add_field_in_tree:
							 | 
						|
								            if self.is_add_field_in_tree:
							 | 
						|
								                inherit_tree_view_name = str(
							 | 
						|
								                    self.tree_view_id.name) + ".inherit.dynamic.custom" + \
							 | 
						|
								                                         str(self.field_description) + ".field"
							 | 
						|
								                optional = "show" if self.is_field_in_tree else "hide"
							 | 
						|
								                tree_view_arch_base = (_(f'''
							 | 
						|
								                    <data>
							 | 
						|
								                        <xpath expr="//field[@name='{self.tree_field_id.name}']" position="{self.tree_field_position}">
							 | 
						|
								                            <field name="{self.name}" optional="{optional}"/>
							 | 
						|
								                        </xpath>
							 | 
						|
								                    </data>'''))
							 | 
						|
								                self.created_tree_view_id = self.env['ir.ui.view'].sudo(
							 | 
						|
								                ).create({
							 | 
						|
								                    'name': inherit_tree_view_name,
							 | 
						|
								                    'type': 'tree',
							 | 
						|
								                    'model': self.model_id.model,
							 | 
						|
								                    'mode': 'extension',
							 | 
						|
								                    'inherit_id': self.tree_view_id.id,
							 | 
						|
								                    'arch_base': tree_view_arch_base,
							 | 
						|
								                    'active': True
							 | 
						|
								                })
							 | 
						|
								                self.write({'status': 'tree'})
							 | 
						|
								                return {
							 | 
						|
								                    'type': 'ir.actions.client',
							 | 
						|
								                    'tag': 'reload',
							 | 
						|
								                }
							 | 
						|
								        else:
							 | 
						|
								            raise ValidationError(
							 | 
						|
								                _('Error! Please select the boolean field Add Field to the '
							 | 
						|
								                  'Tree View.'))
							 | 
						|
								
							 | 
						|
								    @api.onchange('model_id', 'is_add_field_in_tree')
							 | 
						|
								    def set_domain(self):
							 | 
						|
								        """Return the fields that currently present in the form"""
							 | 
						|
								        form_view_ids = self.model_id.view_ids.filtered(
							 | 
						|
								            lambda x: x.type == 'form' and x.mode == 'primary')
							 | 
						|
								        tree_view_ids = self.model_id.view_ids.filtered(
							 | 
						|
								            lambda x: x.type == 'tree' and x.mode == 'primary')
							 | 
						|
								        model_fields = self.env['ir.model.fields'].sudo().search([
							 | 
						|
								            ('model', '=', self.model_id.model)])
							 | 
						|
								        field_list = []
							 | 
						|
								        for rec in model_fields:
							 | 
						|
								            for field in rec:
							 | 
						|
								                field_list.append(field.id)
							 | 
						|
								        return {'domain': {
							 | 
						|
								            'form_view_id': [('id', 'in', form_view_ids.ids)],
							 | 
						|
								            'tree_view_id': [('id', 'in', tree_view_ids.ids)],
							 | 
						|
								            'position_field_id': [('id', 'in', field_list)]
							 | 
						|
								        }}
							 | 
						|
								
							 | 
						|
								    @api.onchange('field_type')
							 | 
						|
								    def onchange_field_type(self):
							 | 
						|
								        """Onchange function of field_type which return domain to widget_id"""
							 | 
						|
								        if self.field_type:
							 | 
						|
								            if self.field_type == 'binary':
							 | 
						|
								                return {'domain': {'widget_id': [('name', '=', 'image')]}}
							 | 
						|
								            elif self.field_type == 'many2many':
							 | 
						|
								                return {'domain': {'widget_id': [
							 | 
						|
								                    ('name', 'in', ['many2many_tags', 'binary'])]}}
							 | 
						|
								            elif self.field_type == 'selection':
							 | 
						|
								                return {'domain': {
							 | 
						|
								                    'widget_id': [('name', 'in', ['radio', 'priority'])]}}
							 | 
						|
								            elif self.field_type == 'float':
							 | 
						|
								                return {'domain': {'widget_id': [('name', '=', 'monetary')]}}
							 | 
						|
								            elif self.field_type == 'many2one':
							 | 
						|
								                return {'domain': {'widget_id': [('name', '=', 'selection')]}}
							 | 
						|
								            else:
							 | 
						|
								                return {'domain': {'widget_id': [('id', '=', False)]}}
							 | 
						|
								        return {'domain': {'widget_id': [('id', '=', False)]}}
							 | 
						|
								
							 | 
						|
								    def unlink(self):
							 | 
						|
								        """unlink function override to make the created form and tree view
							 | 
						|
								        active false"""
							 | 
						|
								        if self.created_form_view_id:
							 | 
						|
								            self.created_form_view_id.active = False
							 | 
						|
								        if self.created_tree_view_id:
							 | 
						|
								            self.created_tree_view_id.active = False
							 | 
						|
								        res = super(DynamicFields, self).unlink()
							 | 
						|
								        return res
							 | 
						|
								
							 |