| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370 | # -*- coding: utf-8 -*-# @authors: Alexander Ezquevo <alexander@acysos.com># Copyright (C) 2015  Acysos S.L.# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl.html).from openerp import models, fields, api, _from openerp.exceptions import Warningfrom datetime import datetimeDFORMAT = "%Y-%m-%d %H:%M:%S"DFORMAT2 = "%Y-%m-%d"class AnimalGroup(models.Model):    _name = 'farm.animal.group'    _order = 'arrival_date desc'    mother = fields.Char(string='Mother', compute='get_mother')    specie = fields.Many2one(comodel_name='farm.specie', string='Specie',                             required=True)    breed = fields.Many2one(comodel_name='farm.specie.breed', string='Breed')    lot = fields.One2many(comodel_name='stock.lot_farm.animal.group',                          inverse_name='animal_group', column1='lot',                          string='Lot')    number = fields.Char(string='Number', compute='get_number', store=True)    location = fields.Many2one(comodel_name='stock.location',                               string='Current location',                               domain=[('usage', '!=', 'view'), ])    farm = fields.Many2one(comodel_name='stock.location',                           string='Current Farm',                           domain=[('usage', '=', 'view')])    quantity = fields.Integer(string='Quantity')    origin = fields.Selection([('purchased', 'Purchased'),                               ('raised', 'Raised'), ], string='Origin',                              required=True,                              default='purchased',                              help='Raised means that this group was born in'                              'the farm. Otherwise, it was purchased.')    arrival_date = fields.Date(string='Arrival Date',                               default=fields.Date.today(),                               help="The date this group arrived (if it was"                               "purchased) or when it was born.")    """    purchase_shipment = fields.Many2one(comodel_name='stock.shipment.in',                                       string='Purchase Shipment',                                       readonly=True)    """    initial_location = fields.Many2one(comodel_name='stock.location',                                       string='Initial location',                                       required=True,                                       domain=[('usage', '=', 'internal'),                                               ('silo', '=', False), ],                                       help="The Location where the group was"                                       "reached or where it was allocated when"                                       "it was purchased.\nIt is used as"                                       "historical information and to get"                                       "Serial Number.")    initial_quantity = fields.Integer(string='Initial quantity', required=True,                                      help="The number of animals in group"                                      "when it was reached or purchased.\nIt"                                      "is used as historical information and"                                      "to create the initial move.")    removal_date = fields.Date(string='Removal date', readonly=True)    weights = fields.One2many(comodel_name='farm.animal.group.weight',                              inverse_name='party', column1='tag',                              string='Weights')    current_weight = fields.Many2one(comodel_name='farm.animal.group.weight',                                     string='Current weight',                                     compute='on_change_with_current_weight')    tags = fields.Many2many(comodel_name='farm.tags',                            inverse_name='animal_group', string='Tag')    notes = fields.Text(string='Notes')    active = fields.Boolean(string='Active', default=True)    feed_quantity = fields.Float(string='cumulative consumed feed')    consumed_feed = fields.Float(string='Consumed Feed per Animal (kg)',                                 compute='get_consumed_feed')    state = fields.Selection(selection=[        ('lactating', 'Lactating'), ('transition', 'Transition'),        ('fatten', 'Faten up'), ('sold', 'Sold')],        readonly=True, default='fatten')    transition_days = fields.Integer(string='Days in Transition',                                     compute='get_transit_days')    fattening_days = fields.Integer(string='Days in fatening',                                    compute='get_fattenig_days')    account = fields.Many2one(comodel_name='account.analytic.account',                              string='Analytic Account')    fatten_date = fields.Date(string='fatten day', compute='_get_fatten_day',                              store=True)    weaning_day = fields.Datetime(string='weaning_day', compute='get_weaning_day')    @api.multi    def get_transformations(self):        tranform_obj = self.env['farm.trasformation.event']        for res in self:            for lot in res.lot:                transforms = tranform_obj.search([                    '|', ('animal_group', '=', res.id),                    ('to_animal_group', '=', )])        @api.multi    def show_feed_event_from_group(self):        feed_ev_obj = self.env['farm.feed.event']        ids = []        for res in self:            for lot in res.lot:                print res.lot                feed_evts = feed_ev_obj.search([                    ('lot', '=', lot.lot.id)])                for event in feed_evts:                    ids.append(event.id)            res = {'view_mode': 'tree,form',                   'res_model': 'farm.feed.event',                   'view_id': False,                   'type': 'ir.actions.act_window',                   'view_type': 'form',                   'domain': [('id', 'in', ids)]}            return res    @api.multi    def get_mother(self):        farrow = self.env['farm.farrowing.event_group']        for res in self:            group_farrow = farrow.search([                ('animal_group', '=', res.id)])            if len(group_farrow)!= 0:                res.mother = group_farrow.event.animal.number            else:                res.mother = '*'    @api.one    def get_weaning_day(self):        if self.state != 'lactating':            weaning_obj = self.env['farm.weaning.event']            wean = weaning_obj.search([                ('farrowing_group', '=', self.id)])            if len(wean) != 0:                self.weaning_day = datetime.strptime(wean.timestamp, DFORMAT)    @api.multi    @api.depends('location', 'state')    def _get_fatten_day(self):        for res in self:            if res.state not in ('lactating', 'transition'):                transformation_obj = self.env['farm.transformation.event']                transition_location = []                for loc in res.specie.lost_found_location:                    transition_location.append(loc.location.id)                transition = transformation_obj.search([                    ('animal_group', '=', res.id),                    ('from_location.id', 'in', transition_location),                    ('to_location.id', 'not in', transition_location)])                if transition:                    res.fatten_date = transition[-1].timestamp                else:                    res.fatten_date = res.arrival_date    @api.one    def get_transit_days(self):        if self.state == 'lactating':            self.transition_days = 0        elif self.state == 'transition':            weaning_obj = self.env['farm.weaning.event']            wean = weaning_obj.search([                ('farrowing_group', '=', self.id)])            if wean:                wean_day = datetime.strptime(wean.timestamp, DFORMAT)                self.transition_days = (datetime.today() - wean_day).days            else:                ref_day = datetime.strptime(self.arrival_date, DFORMAT2)                self.transition_days = (datetime.today() - ref_day).days        else:            weaning_obj = self.env['farm.weaning.event']            transformation_obj = self.env['farm.transformation.event']            wean = weaning_obj.search([                ('farrowing_group', '=', self.id)])            if len(wean) != 0:                wean_day = datetime.strptime(wean.timestamp, DFORMAT)            else:                wean_day = datetime.strptime(self.arrival_date, DFORMAT2)              transition_location = []            for loc in self.specie.lost_found_location:                transition_location.append(loc.location.id)            transition = transformation_obj.search([                ('animal_group', '=', self.id),                ('from_location.id', 'in', transition_location),                ('to_location.id', 'not in', transition_location)])            if transition:                transition_finish = datetime.strptime(                    transition[-1].timestamp, DFORMAT)                self.transition_days = (transition_finish - wean_day).days            else:                self.transition_days = 0    @api.one    def get_fattenig_days(self):        if self.state == 'lactating' or self.state == 'transition':            self.fattening_days = 0        else:            transformation_obj = self.env['farm.transformation.event']            transition_location = []            for loc in self.specie.lost_found_location:                transition_location.append(loc.location.id)            transition = transformation_obj.search([                ('animal_group', '=', self.id),                ('from_location.id', 'in', transition_location),                ('to_location.id', 'not in', transition_location)])            if len(transition) == 0:                transition_finish = datetime.strptime(                    self.arrival_date, '%Y-%m-%d')            else:                transition_finish = datetime.strptime(                    transition[-1].timestamp, DFORMAT)            if self.state == 'fatten':                self.fattening_days = (                    datetime.today() - transition_finish).days            else:                moves_obj = self.env['farm.move.event']                sale_move = moves_obj.search([                    ('animal_group', '=', self.id)])                sale_day = datetime.strptime(                    sale_move[-1].timestamp, DFORMAT)                self.fattening_days = (sale_day - transition_finish).days    @api.multi    def create_first_move(self, res):        moves_obj = self.env['stock.move']        quant_obj = self.env['stock.quant']        for record in res:            if not record.lot:                        production_lot_obj = self.env['stock.production.lot']                        animal_group_lot_obj = \                            self.env['stock.lot_farm.animal.group']                        new_lot = production_lot_obj.create({                            'product_id': res.specie.group_product.id,                            'animal_type': 'group',                            })                        animal_group_lot_obj.create({                            'lot': new_lot.id,                            'animal_group': res.id})            quant = quant_obj.search([                ('lot_id', '=', record.lot[0].lot.id),                ('location_id', '=', record.initial_location.id)                ])            record.location = record.initial_location            if len(record.lot) > 1:                    raise Warning(                        _('lots can not be mixed in an initial group, create a'                          ' group for each lot and then group them into the'                          ' desired group'))            elif record.origin == 'raised':                if not quant:                    raise_location = self.env['stock.location'].search(                        [('usage', '=', 'production')])                    uom = record.lot.lot.product_id.product_tmpl_id.uom_id.id                    new_move = moves_obj.create({                        'name': 'raise-' + record.lot[0].lot.name,                        'create_date': fields.Date.today(),                        'date': record.arrival_date,                        'product_id': record.lot[0].lot.product_id.id,                        'product_uom_qty': record.initial_quantity,                        'product_uom': uom,                        'location_id': raise_location.id,                        'location_dest_id': record.initial_location.id,                        'company_id': record.initial_location.company_id.id,                        })                    new_move.action_done()                    new_move.quant_ids.lot_id = record.lot[0].lot.id                else:                    raise Warning(                        _('this lot iis in use, please create new lot'))            else:                if not quant:                    raise Warning(                        _('no product in farms for this lot'))                target_quant = False                for q in quant:                    if q.location_id == record.initial_location:                        if q.qty >= record.initial_quantity:                            target_quant = q                if not target_quant:                    raise Warning(                        _('group intial quantity and product quantity '                          'are diferent'))                an_group = self.env['farm.animal.group'].search([                    ('lot.lot.id', '=', record.lot.lot.id),                    ('id', '!=', record.id)])                if len(an_group) > 0:                    raise Warning(                        _('this lot is in use from oder group'))    @api.model    @api.returns('self', lambda value: value.id)    def create(self, vals):        res = super(AnimalGroup, self).create(vals)        self.create_first_move(res)        res.quantity = res.initial_quantity        analy_ac_obj = self.env['account.analytic.account']        top_account = analy_ac_obj.search([            ('name', '=', res.farm.name)])        if not top_account:            gen_account = analy_ac_obj.search([                ('name', '=', 'General Account')])            if not gen_account:                gen_account = analy_ac_obj.create({'name': 'General Account'})            top_account = analy_ac_obj.create({'name': res.farm.name,                                               'parent_id': gen_account.id})        new_account = analy_ac_obj.create({            'name': 'AA-group-'+res.number,            'parent_id': top_account.id})        res.account = new_account        return res    @api.multi    def name_get(self):        result = ''        displayName = []        for group in self:            if group.tags:                displayName.append(                    (group.id, group.number + '-' +group.tags[0].name))            else:                displayName.append((group.id, group.number))        return displayName    @api.multi    @api.depends('lot')    def get_number(self):        for group in self:            result = '*'            if len(group.lot) > 2:                result = group.lot[2].lot.name            elif len(group.lot) > 0:                result = group.lot[0].lot.name            group.number = result    def get_locations(self):        return False    @api.one    def on_change_with_current_weight(self):        if self.weights:            self.current_weight = self.weights[0].id    @api.one    def get_consumed_feed(self):        if self.quantity == 0:            self.consumed_feed = self.feed_quantity/self.initial_quantity        else:            self.consumed_feed = self.feed_quantity/self.quantityclass AnimalGroupWeight(models.Model):    _name = 'farm.animal.group.weight'    _order = 'timestamp DESC'    rec_name = 'weight'    party = fields.Many2one(comodel_name='farm.animal.group',                            string='Group', ondelete='CASCADE',                            required=True)    timestamp = fields.Datetime(string='Date & time',                                default=fields.Datetime.now())    quantity = fields.Integer(string='Number of individuals', required=True)    uom = fields.Many2one(comodel_name='product.uom', string='Uom')    weight = fields.Float(string='Weihht', digits=(3, 2), required=True)    @api.onchange('timestamp')    def get_defaults(self):        if self.party is not False:            self.quantity = self.party.quantity
 |