Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Test commit & push MEPL #166

Open
wants to merge 6 commits into
base: 18.0
Choose a base branch
from

Conversation

Mathilde411
Copy link

No description provided.

@Mathilde411 Mathilde411 force-pushed the 18.0-tutorials-mepl branch 2 times, most recently from 9119b6c to 11ab1a4 Compare October 22, 2024 12:59
@Mathilde411
Copy link
Author

Mathilde411 commented Oct 22, 2024

Sorry @aboo-odoo I force-pushed over you because I did not see you commited :x

@proose
Copy link

proose commented Oct 22, 2024

Hey @Mathilde411, no worries, I'll push it again on your branch (but be careful when working with someone in the future, as you might delete his/her work). Go on with your tutorial for now and when you'll want to push, try to fetch this first (and make it work). 👍 😄

@Mathilde411
Copy link
Author

Mathilde411 commented Oct 22, 2024

Hey @Mathilde411, no worries, I'll push it again on your branch (but be careful when working with someone in the future, as you might delete his/her work). Go on with your tutorial for now and when you'll want to push, try to fetch this first (and make it work). 👍 😄

Yeah, I wasn't really thinking that someone else could have pushed on my branch right now ^^
I'll be more careful next time :)

@proose
Copy link

proose commented Oct 22, 2024

Yeah, I wasn't really thinking that someone else could have pushed on my branch right now ^^
I'll be more careful next time :)

People should warn you beforehand, I was just testing you a bit 😁
I would suggest you use a --force-with-lease instead of a --force with your push command, to avoid this kind of issue (more info at https://stackoverflow.com/questions/52823692/git-push-force-with-lease-vs-force) :)

Added Real estate app built up to the end of chapter 11
Chapter 12 inheritance
Diverged a bit from chapter. Digged into sales order code to adapt it while really understanding instead of following tutorial
Copy link

@proose proose left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

A bit of nitpicks, but great job in general !
Very well done !

('check_bedrooms', 'CHECK(bedrooms >= 0)', 'The number of bedrooms must be positive.'),
('check_living_area', 'CHECK(living_area > 0)', 'The living area must be positive.'),
('check_facades', 'CHECK(facades > 0)', 'The number of facades must be positive.'),
('check_name_unique', 'UNIQUE(name)', 'The Property name must be unique.')
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This constraint was not asked by the tutorial. You can add it if you feel it is better, as it is a tutorial, but you might want to adapt the corresponding field... As of now, duplicating the property raises an error.

Comment on lines +81 to +83
if not float_is_zero(record.selling_price, precision_digits=2) and float_compare(record.selling_price,
0.9 * record.expected_price,
precision_digits=2) < 0:
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
if not float_is_zero(record.selling_price, precision_digits=2) and float_compare(record.selling_price,
0.9 * record.expected_price,
precision_digits=2) < 0:
if (
not float_is_zero(record.selling_price, precision_digits=2)
and float_compare(record.selling_price, 0.9 * record.expected_price, precision_digits=2) < 0
):

This is overall preferable and more clear (nitpicky, I admit) :D

partner_id = fields.Many2one('res.partner', string='Partner', required=True)
price = fields.Float()
property_id = fields.Many2one('estate.property', string='Property', required=True)
create_date = fields.Date(string='Create Date', default=lambda self: fields.Datetime.now())
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

create_date is an automatic field and should not be added here (see https://www.odoo.com/documentation/18.0/developer/tutorials/server_framework_101/03_basicmodel.html#:~:text=of%20the%20model.-,create_date,-().

I see you're already using fields.Datetime.now(): you can just use it instead of create_date if create_date is not set, in the relevant methods :)

for record in self:
record.deadline_date = record.create_date + relativedelta(days=record.validity)

@api.depends('validity', 'deadline_date', 'create_date')
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This is not needed for an inverse method (I guess it was just copy-pasted from above)

Comment on lines +40 to +59
def action_accept(self, propagate=True):
for record in self:
if propagate:
record.property_id.set_accepted_offer(record)
record.state = 'accepted'
return True

def action_refuse(self, propagate=True):
for record in self:
if record.state == 'accepted' and propagate:
record.property_id.set_accepted_offer(None)
record.state = 'refused'
return True

def action_reset(self, propagate=True):
for record in self:
if record.state == 'accepted' and propagate:
record.property_id.set_accepted_offer(None)
record.state = 'received'
return True
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It's better to bring all code about set_accepted_offer in this model and remove it from estate.property as the action comes from here

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I was doing this because of ownership. Isn't it better to let each class manage its own variables ?

Copy link

@proose proose Oct 25, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

While it can be true in "classic" frameworks, in Odoo, a lot of modules depend on each other, and might want to adapt the behavior of the inherited model, or add features. If we add everything in the inherited class, it would become monstrously huge and hard to work in.

In that sense, we can apply the same thinking with files within a same module: as the "offer acceptance feature" comes from this file, it can handle the property variables by itself. We don't usually use setters and getters unless there are strong conditions to respect.

At the end of the day, it is just a (heavy) preference and your way of doing might be accepted, I was particularly nitpicky because your implem was very well done 😄

_inherit = 'res.users'

property_ids = fields.One2many('estate.property', 'salesperson_id', string='Properties',
domain="[('date_availability', '&lt;=', context_today().strftime('%Y-%m-%d'))]")
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
domain="[('date_availability', '&lt;=', context_today().strftime('%Y-%m-%d'))]")
domain="[('date_availability', '<=', fields.Date.today())])

It doesn't work the way you did: you can try yourself by going to the Setting App -> Users & Companies and looking into the user you set as salesperson (with unavailable properties set up for him): you'll see that even unavailable properties are still shown by default

</record>

<record id="estate_property_offer_action" model="ir.actions.act_window">
<field name="name">Property Types</field>
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Forgot to change the name here ? :D

<field name="arch" type="xml">
<form string="Property Types">
<sheet>
<div class="oe_button_box" name="button_box">
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This button_box div is not needed IMO

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Isn't it like that to add smart buttons ? I saw it in the sales module.

Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Have you tried without ? I think this should work with just the inside node (<button class="oe_stat_button"...), if not, you can ignore my comment

Comment on lines +41 to +43
<div class="oe_button_box" name="button_box">

</div>
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Forgot to remove ?

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It is here to let modules add box buttons without stepping on each other.
Ex: If 2 modules add buttons they already have the parent div, and do not need to step on each other to add it

Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

That is good thinking if you know it will be used for inheritance ! But usually you never know if that will be the case, and we avoid adding extra nodes like this unless we know this part will particularly be inherited multiple times. Or if it can't be done otherwise, but here you can use expr="//div[@class='oe_title']" in your inheriting view to get it.

In this case I insists because I don't think an extra <div class="oe_button_box" name="button_box"> is needed around the oe-stat button, but if it is, you can eventually leave it like this

'version': '1.0',
'category': 'Tutorials',
'depends': [
'base',
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This is not needed as the inheritance path of account already include base

Copy link

@proose proose left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

A bit more nitpicks :D

More hints: usually we try to keep the file names and class names close to the model name: model 'estate.property' -> file 'estate_property.py' and class 'EstatePropery'. This allow the devs to more easily search the file we need and it appears more clear when looking into files with multiple models inside.

It's a good thing to go look into what's existing in Odoo - you'll learn a lot. But it is usefull to look at different implementations of what you're looking for: some modules are heavy in processing while others might be way more simple and quick... It depends on what's needed 🤷‍♂️ .

Very good work so far, you're quick to learn !

Comment on lines +15 to +20
# invoice_status = fields.Selection([
# ('upselling', 'Upselling Opportunity'),
# ('invoiced', 'Fully Invoiced'),
# ('to invoice', 'To Invoice'),
# ('no', 'Nothing to Invoice')
# ], string='Invoice Status', compute='_get_invoice_status', store=True, readonly=True)
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Don't forget to remove your old comments


res = {
'display_type': 'product',
'sequence': sequence,
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The sequence is already automatically handled, no use adding it here


# trying to understand sales invoice creation line by line
def _create_invoices(self, grouped=False, date=None):
if not self.env['account.move'].has_access('create'):
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The ORM handles access errors by itself, you can keep it simpler :D

Comment on lines +71 to +117
invoice_vals_list = []
for invoice_item_sequence, record in enumerate(self):
if record.state != 'sold':
raise UserError('You can only invoice sold properties.')

if record.buyer_id.lang:
record = record.with_context(lang=record.buyer_id.lang)

invoice_vals = record._prepare_invoice()
invoice_vals['invoice_line_ids'].append(
fields.Command.create(record._prepare_invoice_line(sequence=invoice_item_sequence))
)
invoice_vals_list.append(invoice_vals)

if not grouped:
new_invoice_vals_list = []
invoice_grouping_keys = self._get_invoice_grouping_keys()
invoice_vals_list = sorted(
invoice_vals_list,
key=lambda x: [
x.get(grouping_key) for grouping_key in invoice_grouping_keys
]
)

for _grouping_keys, invoices in groupby(invoice_vals_list,
key=lambda x: (x.get(grouping_key) for grouping_key in
invoice_grouping_keys)):
origins = set()
refs = set()
ref_invoice_vals = None
for invoice_vals in invoices:
if not ref_invoice_vals:
ref_invoice_vals = invoice_vals
else:
ref_invoice_vals['invoice_line_ids'] += invoice_vals['invoice_line_ids']
origins.add(invoice_vals['invoice_origin'])
refs.add(invoice_vals['ref'])

ref_invoice_vals.update({
'ref': ', '.join(refs)[:2000],
'invoice_origin': ', '.join(origins)
})
new_invoice_vals_list.append(ref_invoice_vals)
invoice_vals_list = new_invoice_vals_list

return self.env['account.move'].sudo().with_context(default_move_type='out_invoice').create(
invoice_vals_list)
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This whole method is way too complicated for our case. You can practically use the same example as shown in the tutorial to create a move with the correct invoice_line_ids ;)

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I was thinkig it'd be cool to keep the possibility of creating 1 invoice for multiple properties :D

Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

That would be nice indeed. You can then use a small method that can add a invoice line to an existing move, or put that simple code directly into the method where you create the move (assuming you change a bit the flow to allow creating a move after selecting multiple properties for example), but this code is still doing way too complicated (and useless in our case) things :P

Comment on lines +120 to +128
action = self.env['ir.actions.actions']._for_xml_id('account.action_move_out_invoice_type')
form_view = [(self.env.ref('account.view_move_form').id, 'form')]
if 'views' in action:
action['views'] = form_view + [(state, view) for state, view in action['views'] if view != 'form']
else:
action['views'] = form_view
action['res_id'] = self.invoice_id.id

return action
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This is also too complicated with what is needed. In python, you can return a dict to communicate an action (unless the method is called from a simple other python method). Your case should not need more than this (you'll probably want to adapt it to make it work):

Suggested change
action = self.env['ir.actions.actions']._for_xml_id('account.action_move_out_invoice_type')
form_view = [(self.env.ref('account.view_move_form').id, 'form')]
if 'views' in action:
action['views'] = form_view + [(state, view) for state, view in action['views'] if view != 'form']
else:
action['views'] = form_view
action['res_id'] = self.invoice_id.id
return action
return {
'name': 'Invoice',
'view_mode': 'form',
'res_model': 'account.move',
'type': 'ir.actions.act_window',
'res_id': self.invoice_id.id,
}

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

2 participants