[MERGE] forward port of branch saas-3 up to 4601d85

This commit is contained in:
Christophe Simonis 2014-06-10 12:29:47 +02:00
commit 6b8e9727e5
18 changed files with 72 additions and 54 deletions

View File

@ -77,7 +77,7 @@ class account_analytic_invoice_line(osv.osv):
price = price_unit price = price_unit
elif pricelist_id: elif pricelist_id:
price = res.price price = res.price
else: if price is False:
price = res.list_price price = res.list_price
if not name: if not name:
name = self.pool.get('product.product').name_get(cr, uid, [res.id], context=local_context)[0][1] name = self.pool.get('product.product').name_get(cr, uid, [res.id], context=local_context)[0][1]

View File

@ -139,7 +139,8 @@
<group> <group>
<field name="pricelist_id" <field name="pricelist_id"
class="oe_inline" class="oe_inline"
attrs="{'required': [('invoice_on_timesheets', '=', True)]}"/> attrs="{'required': [('invoice_on_timesheets', '=', True)]}"
domain="[('type', '=', 'sale')]"/>
<field name="to_invoice" <field name="to_invoice"
class="oe_inline" class="oe_inline"
widget="selection" widget="selection"

View File

@ -124,7 +124,7 @@ class pos_make_payment(osv.osv_memory):
} }
_defaults = { _defaults = {
'journal_id' : _default_journal, 'journal_id' : _default_journal,
'payment_date': time.strftime('%Y-%m-%d %H:%M:%S'), 'payment_date': lambda *a: time.strftime('%Y-%m-%d %H:%M:%S'),
'amount': _default_amount, 'amount': _default_amount,
} }

View File

@ -198,7 +198,8 @@ class project_work(osv.osv):
if amount_unit and 'amount' in amount_unit.get('value',{}): if amount_unit and 'amount' in amount_unit.get('value',{}):
vals_line['amount'] = amount_unit['value']['amount'] vals_line['amount'] = amount_unit['value']['amount']
self.pool.get('hr.analytic.timesheet').write(cr, uid, [line_id.id], vals_line, context=context) if vals_line:
self.pool.get('hr.analytic.timesheet').write(cr, uid, [line_id.id], vals_line, context=context)
return super(project_work,self).write(cr, uid, ids, vals, context) return super(project_work,self).write(cr, uid, ids, vals, context)

View File

@ -37,6 +37,7 @@ from openerp import report
import tempfile import tempfile
import time import time
import logging import logging
from functools import partial
from report_helper import WebKitHelper from report_helper import WebKitHelper
import openerp import openerp
@ -119,7 +120,6 @@ class WebKitParser(report_sxw):
""" """
def __init__(self, name, table, rml=False, parser=rml_parse, def __init__(self, name, table, rml=False, parser=rml_parse,
header=True, store=False, register=True): header=True, store=False, register=True):
self.parser_instance = False
self.localcontext = {} self.localcontext = {}
report_sxw.__init__(self, name, table, rml, parser, report_sxw.__init__(self, name, table, rml, parser,
header, store, register=register) header, store, register=register)
@ -230,16 +230,16 @@ class WebKitParser(report_sxw):
_logger.error('cannot remove file %s: %s', f_to_del, exc) _logger.error('cannot remove file %s: %s', f_to_del, exc)
return pdf return pdf
def translate_call(self, src): def translate_call(self, parser_instance, src):
"""Translate String.""" """Translate String."""
ir_translation = self.pool['ir.translation'] ir_translation = self.pool['ir.translation']
name = self.tmpl and 'addons/' + self.tmpl or None name = self.tmpl and 'addons/' + self.tmpl or None
res = ir_translation._get_source(self.parser_instance.cr, self.parser_instance.uid, res = ir_translation._get_source(parser_instance.cr, parser_instance.uid,
name, 'report', self.parser_instance.localcontext.get('lang', 'en_US'), src) name, 'report', parser_instance.localcontext.get('lang', 'en_US'), src)
if res == src: if res == src:
# no translation defined, fallback on None (backward compatibility) # no translation defined, fallback on None (backward compatibility)
res = ir_translation._get_source(self.parser_instance.cr, self.parser_instance.uid, res = ir_translation._get_source(parser_instance.cr, parser_instance.uid,
None, 'report', self.parser_instance.localcontext.get('lang', 'en_US'), src) None, 'report', parser_instance.localcontext.get('lang', 'en_US'), src)
if not res : if not res :
return src return src
return res return res
@ -264,14 +264,14 @@ class WebKitParser(report_sxw):
if report_xml.report_type != 'webkit': if report_xml.report_type != 'webkit':
return super(WebKitParser,self).create_single_pdf(cursor, uid, ids, data, report_xml, context=context) return super(WebKitParser,self).create_single_pdf(cursor, uid, ids, data, report_xml, context=context)
self.parser_instance = self.parser(cursor, parser_instance = self.parser(cursor,
uid, uid,
self.name2, self.name2,
context=context) context=context)
self.pool = pool self.pool = pool
objs = self.getObjects(cursor, uid, ids, context) objs = self.getObjects(cursor, uid, ids, context)
self.parser_instance.set_context(objs, data, ids, report_xml.report_type) parser_instance.set_context(objs, data, ids, report_xml.report_type)
template = False template = False
@ -299,21 +299,22 @@ class WebKitParser(report_sxw):
if not css : if not css :
css = '' css = ''
translate_call = partial(self.translate_call, parser_instance)
body_mako_tpl = mako_template(template) body_mako_tpl = mako_template(template)
helper = WebKitHelper(cursor, uid, report_xml.id, context) helper = WebKitHelper(cursor, uid, report_xml.id, context)
self.parser_instance.localcontext['helper'] = helper parser_instance.localcontext['helper'] = helper
self.parser_instance.localcontext['css'] = css parser_instance.localcontext['css'] = css
self.parser_instance.localcontext['_'] = self.translate_call parser_instance.localcontext['_'] = translate_call
# apply extender functions # apply extender functions
additional = {} additional = {}
if xml_id in _extender_functions: if xml_id in _extender_functions:
for fct in _extender_functions[xml_id]: for fct in _extender_functions[xml_id]:
fct(pool, cr, uid, self.parser_instance.localcontext, context) fct(pool, cr, uid, parser_instance.localcontext, context)
if report_xml.precise_mode: if report_xml.precise_mode:
ctx = dict(self.parser_instance.localcontext) ctx = dict(parser_instance.localcontext)
for obj in self.parser_instance.localcontext['objects']: for obj in parser_instance.localcontext['objects']:
ctx['objects'] = [obj] ctx['objects'] = [obj]
try : try :
html = body_mako_tpl.render(dict(ctx)) html = body_mako_tpl.render(dict(ctx))
@ -324,7 +325,7 @@ class WebKitParser(report_sxw):
raise except_osv(_('Webkit render!'), msg) raise except_osv(_('Webkit render!'), msg)
else: else:
try : try :
html = body_mako_tpl.render(dict(self.parser_instance.localcontext)) html = body_mako_tpl.render(dict(parser_instance.localcontext))
htmls.append(html) htmls.append(html)
except Exception, e: except Exception, e:
msg = u"%s" % e msg = u"%s" % e
@ -332,22 +333,21 @@ class WebKitParser(report_sxw):
raise except_osv(_('Webkit render!'), msg) raise except_osv(_('Webkit render!'), msg)
head_mako_tpl = mako_template(header) head_mako_tpl = mako_template(header)
try : try :
head = head_mako_tpl.render(dict(self.parser_instance.localcontext, _debug=False)) head = head_mako_tpl.render(dict(parser_instance.localcontext, _debug=False))
except Exception, e: except Exception, e:
raise except_osv(_('Webkit render!'), u"%s" % e) raise except_osv(_('Webkit render!'), u"%s" % e)
foot = False foot = False
if footer : if footer :
foot_mako_tpl = mako_template(footer) foot_mako_tpl = mako_template(footer)
try : try :
foot = foot_mako_tpl.render(dict({}, foot = foot_mako_tpl.render(dict(parser_instance.localcontext))
**self.parser_instance.localcontext))
except Exception, e: except Exception, e:
msg = u"%s" % e msg = u"%s" % e
_logger.error(msg) _logger.error(msg)
raise except_osv(_('Webkit render!'), msg) raise except_osv(_('Webkit render!'), msg)
if report_xml.webkit_debug : if report_xml.webkit_debug :
try : try :
deb = head_mako_tpl.render(dict(self.parser_instance.localcontext, _debug=tools.ustr("\n".join(htmls)))) deb = head_mako_tpl.render(dict(parser_instance.localcontext, _debug=tools.ustr("\n".join(htmls))))
except Exception, e: except Exception, e:
msg = u"%s" % e msg = u"%s" % e
_logger.error(msg) _logger.error(msg)

View File

@ -1126,13 +1126,13 @@ class stock_picking(osv.osv):
if not inv_type: if not inv_type:
inv_type = self._get_invoice_type(picking) inv_type = self._get_invoice_type(picking)
invoice_vals = self._prepare_invoice(cr, uid, picking, partner, inv_type, journal_id, context=context)
if group and partner.id in invoices_group: if group and partner.id in invoices_group:
invoice_id = invoices_group[partner.id] invoice_id = invoices_group[partner.id]
invoice = invoice_obj.browse(cr, uid, invoice_id) invoice = invoice_obj.browse(cr, uid, invoice_id)
invoice_vals_group = self._prepare_invoice_group(cr, uid, picking, partner, invoice, context=context) invoice_vals_group = self._prepare_invoice_group(cr, uid, picking, partner, invoice, context=context)
invoice_obj.write(cr, uid, [invoice_id], invoice_vals_group, context=context) invoice_obj.write(cr, uid, [invoice_id], invoice_vals_group, context=context)
else: else:
invoice_vals = self._prepare_invoice(cr, uid, picking, partner, inv_type, journal_id, context=context)
invoice_id = invoice_obj.create(cr, uid, invoice_vals, context=context) invoice_id = invoice_obj.create(cr, uid, invoice_vals, context=context)
invoices_group[partner.id] = invoice_id invoices_group[partner.id] = invoice_id
res[picking.id] = invoice_id res[picking.id] = invoice_id
@ -2188,12 +2188,12 @@ class stock_move(osv.osv):
done.append(move.id) done.append(move.id)
pickings[move.picking_id.id] = 1 pickings[move.picking_id.id] = 1
r = res.pop(0) r = res.pop(0)
product_uos_qty = self.pool.get('stock.move').onchange_quantity(cr, uid, ids, move.product_id.id, r[0], move.product_id.uom_id.id, move.product_id.uos_id.id)['value']['product_uos_qty'] product_uos_qty = self.pool.get('stock.move').onchange_quantity(cr, uid, [move.id], move.product_id.id, r[0], move.product_id.uom_id.id, move.product_id.uos_id.id)['value']['product_uos_qty']
cr.execute('update stock_move set location_id=%s, product_qty=%s, product_uos_qty=%s where id=%s', (r[1], r[0],product_uos_qty, move.id)) cr.execute('update stock_move set location_id=%s, product_qty=%s, product_uos_qty=%s where id=%s', (r[1], r[0],product_uos_qty, move.id))
while res: while res:
r = res.pop(0) r = res.pop(0)
product_uos_qty = self.pool.get('stock.move').onchange_quantity(cr, uid, ids, move.product_id.id, r[0], move.product_id.uom_id.id, move.product_id.uos_id.id)['value']['product_uos_qty'] product_uos_qty = self.pool.get('stock.move').onchange_quantity(cr, uid, [move.id], move.product_id.id, r[0], move.product_id.uom_id.id, move.product_id.uos_id.id)['value']['product_uos_qty']
move_id = self.copy(cr, uid, move.id, {'product_uos_qty': product_uos_qty, 'product_qty': r[0], 'location_id': r[1]}) move_id = self.copy(cr, uid, move.id, {'product_uos_qty': product_uos_qty, 'product_qty': r[0], 'location_id': r[1]})
done.append(move_id) done.append(move_id)
if done: if done:

View File

@ -4732,7 +4732,8 @@ instance.web.form.Many2ManyListView = instance.web.ListView.extend(/** @lends in
pop.select_element( pop.select_element(
this.model, this.model,
{ {
title: _t("Add: ") + this.m2m_field.string title: _t("Add: ") + this.m2m_field.string,
no_create: this.m2m_field.options.no_create,
}, },
new instance.web.CompoundDomain(this.m2m_field.build_domain(), ["!", ["id", "in", this.m2m_field.dataset.ids]]), new instance.web.CompoundDomain(this.m2m_field.build_domain(), ["!", ["id", "in", this.m2m_field.dataset.ids]]),
this.m2m_field.build_context() this.m2m_field.build_context()

View File

@ -1472,8 +1472,9 @@
<t t-if="! widget.options.disable_multiple_selection"> <t t-if="! widget.options.disable_multiple_selection">
<button type="button" class="oe_button oe_selectcreatepopup-search-select oe_highlight" disabled="disabled">Select</button> <button type="button" class="oe_button oe_selectcreatepopup-search-select oe_highlight" disabled="disabled">Select</button>
</t> </t>
<t t-if="!widget.options.no_create">
<button type="button" class="oe_button oe_selectcreatepopup-search-create">Create</button> <button type="button" class="oe_button oe_selectcreatepopup-search-create">Create</button>
or <a class="oe_selectcreatepopup-search-close oe_bold oe_form_button_cancel" href="javascript:void(0)">Cancel</a> or </t><a class="oe_selectcreatepopup-search-close oe_bold oe_form_button_cancel" href="javascript:void(0)">Cancel</a>
</t> </t>
<t t-name="AbstractFormPopup.buttons"> <t t-name="AbstractFormPopup.buttons">
<t t-if="! readonly"> <t t-if="! readonly">

View File

@ -95,6 +95,9 @@
.openerp .oe_kanban_view .oe_kanban_groups { .openerp .oe_kanban_view .oe_kanban_groups {
height: inherit; height: inherit;
} }
.openerp .oe_kanban_view .oe_kanban_groups_records {
height: 100%;
}
.openerp .oe_kanban_view.oe_kanban_grouped_by_m2o .oe_kanban_group_title { .openerp .oe_kanban_view.oe_kanban_grouped_by_m2o .oe_kanban_group_title {
cursor: move; cursor: move;
} }
@ -162,10 +165,7 @@
padding: 0px; padding: 0px;
background: white; background: white;
} }
.openerp .oe_kanban_view .oe_kanban_column, .openerp .oe_kanban_view .oe_kanban_column_cards { .openerp .oe_kanban_view.oe_kanban_grouped .oe_kanban_column, .openerp .oe_kanban_view.oe_kanban_grouped .oe_kanban_column_cards {
height: 100%;
}
.openerp .oe_kanban_view .oe_kanban_column, .openerp .oe_kanban_view .oe_kanban_column_cards {
height: 100%; height: 100%;
} }
.openerp .oe_kanban_view .oe_kanban_aggregates { .openerp .oe_kanban_view .oe_kanban_aggregates {

View File

@ -119,6 +119,8 @@
// KanbanGroups {{{ // KanbanGroups {{{
.oe_kanban_groups .oe_kanban_groups
height: inherit height: inherit
.oe_kanban_groups_records
height: 100%
&.oe_kanban_grouped_by_m2o .oe_kanban_group_title &.oe_kanban_grouped_by_m2o .oe_kanban_group_title
cursor: move cursor: move
.oe_kanban_header .oe_kanban_header
@ -178,9 +180,9 @@
.oe_kanban_column .oe_kanban_column
padding: 0px padding: 0px
background: #ffffff background: #ffffff
&.oe_kanban_grouped
.oe_kanban_column, .oe_kanban_column_cards .oe_kanban_column, .oe_kanban_column_cards
height: 100% height: 100%
.oe_kanban_aggregates .oe_kanban_aggregates
padding: 0 padding: 0
margin: 0px margin: 0px

View File

@ -973,7 +973,9 @@
}, },
clean_for_save: function () { clean_for_save: function () {
this._super(); this._super();
this.$target.find(".item").removeClass("next prev left right"); this.$target.find(".item").removeClass("next prev left right active");
this.$indicators.find('li').removeClass('active');
this.$indicators.find('li:first').addClass('active');
if(!this.$target.find(".item.active").length) { if(!this.$target.find(".item.active").length) {
this.$target.find(".item:first").addClass("active"); this.$target.find(".item:first").addClass("active");
} }

View File

@ -209,7 +209,8 @@
e=o.createElement(i);r=o.getElementsByTagName(i)[0]; e=o.createElement(i);r=o.getElementsByTagName(i)[0];
e.src='//www.google-analytics.com/analytics.js'; e.src='//www.google-analytics.com/analytics.js';
r.parentNode.insertBefore(e,r)}(window,document,'script','ga')); r.parentNode.insertBefore(e,r)}(window,document,'script','ga'));
ga('create','<t t-esc="website.google_analytics_key"/>');ga('send','pageview'); ga('create',_.str.trim('<t t-esc="website.google_analytics_key"/>'));
ga('send','pageview');
</script> </script>
</t> </t>
</body> </body>

View File

@ -134,9 +134,7 @@
</h3> </h3>
<ul class="list-inline mb0"> <ul class="list-inline mb0">
<li t-if="track.speaker_ids" class="text-muted fa fa-user"> <li t-if="track.speaker_ids" class="text-muted fa fa-user">
<t t-foreach="track.speaker_ids" t-as="speaker"> <t t-esc="', '.join([speaker.name for speaker in track.speaker_ids])"></t>
<t t-esc="speaker.name"/>,
</t>
</li> </li>
<li class="text-muted fa fa-calendar"> <span t-field="track.date" t-field-options='{"hide_seconds":"True"}'/></li> <li class="text-muted fa fa-calendar"> <span t-field="track.date" t-field-options='{"hide_seconds":"True"}'/></li>
<li class="text-muted fa fa-map-marker" t-if="track.location_id"> <li class="text-muted fa fa-map-marker" t-if="track.location_id">

View File

@ -238,7 +238,7 @@ class ir_cron(osv.osv):
locked_job = lock_cr.fetchone() locked_job = lock_cr.fetchone()
if not locked_job: if not locked_job:
# job was already executed by another parallel process/thread, skipping it. _logger.debug("Job `%s` already executed by another process/thread. skipping it", job['name'])
continue continue
# Got the lock on the job row, run its code # Got the lock on the job row, run its code
_logger.debug('Starting job `%s`.', job['name']) _logger.debug('Starting job `%s`.', job['name'])

View File

@ -1141,13 +1141,20 @@ class ir_model_data(osv.osv):
# Remove non-model records first, then model fields, and finish with models # Remove non-model records first, then model fields, and finish with models
unlink_if_refcount((model, res_id) for model, res_id in to_unlink unlink_if_refcount((model, res_id) for model, res_id in to_unlink
if model not in ('ir.model','ir.model.fields')) if model not in ('ir.model','ir.model.fields','ir.model.constraint'))
unlink_if_refcount((model, res_id) for model, res_id in to_unlink
if model == 'ir.model.constraint')
ir_module_module = self.pool['ir.module.module']
ir_model_constraint = self.pool['ir.model.constraint']
modules_to_remove_ids = ir_module_module.search(cr, uid, [('name', 'in', modules_to_remove)], context=context)
constraint_ids = ir_model_constraint.search(cr, uid, [('module', 'in', modules_to_remove_ids)], context=context)
ir_model_constraint._module_data_uninstall(cr, uid, constraint_ids, context)
unlink_if_refcount((model, res_id) for model, res_id in to_unlink unlink_if_refcount((model, res_id) for model, res_id in to_unlink
if model == 'ir.model.fields') if model == 'ir.model.fields')
ir_model_relation = self.pool['ir.model.relation'] ir_model_relation = self.pool['ir.model.relation']
ir_module_module = self.pool['ir.module.module']
modules_to_remove_ids = ir_module_module.search(cr, uid, [('name', 'in', modules_to_remove)])
relation_ids = ir_model_relation.search(cr, uid, [('module', 'in', modules_to_remove_ids)]) relation_ids = ir_model_relation.search(cr, uid, [('module', 'in', modules_to_remove_ids)])
ir_model_relation._module_data_uninstall(cr, uid, relation_ids, context) ir_model_relation._module_data_uninstall(cr, uid, relation_ids, context)

View File

@ -436,11 +436,7 @@ class module(osv.osv):
including the deletion of all database structures created by the module: including the deletion of all database structures created by the module:
tables, columns, constraints, etc.""" tables, columns, constraints, etc."""
ir_model_data = self.pool.get('ir.model.data') ir_model_data = self.pool.get('ir.model.data')
ir_model_constraint = self.pool.get('ir.model.constraint')
modules_to_remove = [m.name for m in self.browse(cr, uid, ids, context)] modules_to_remove = [m.name for m in self.browse(cr, uid, ids, context)]
modules_to_remove_ids = [m.id for m in self.browse(cr, uid, ids, context)]
constraint_ids = ir_model_constraint.search(cr, uid, [('module', 'in', modules_to_remove_ids)])
ir_model_constraint._module_data_uninstall(cr, uid, constraint_ids, context)
ir_model_data._module_data_uninstall(cr, uid, modules_to_remove, context) ir_model_data._module_data_uninstall(cr, uid, modules_to_remove, context)
self.write(cr, uid, ids, {'state': 'uninstalled'}) self.write(cr, uid, ids, {'state': 'uninstalled'})
return True return True

View File

@ -1261,8 +1261,16 @@ class function(_column):
return new_values return new_values
def get(self, cr, obj, ids, name, uid=False, context=None, values=None): def get(self, cr, obj, ids, name, uid=False, context=None, values=None):
result = self._fnct(obj, cr, uid, ids, name, self._arg, context) multi = self._multi
if self._multi: # if we already have a value, don't recompute it.
# This happen if case of stored many2one fields
if values and not multi and name in values[0]:
result = {v['id']: v[name] for v in values}
elif values and multi and all(n in values[0] for n in name):
result = {v['id']: dict({n: v[n]} for n in name) for v in values}
else:
result = self._fnct(obj, cr, uid, ids, name, self._arg, context)
if multi:
swap = {} swap = {}
for rid, values in result.iteritems(): for rid, values in result.iteritems():
for f, v in values.iteritems(): for f, v in values.iteritems():

View File

@ -2816,7 +2816,7 @@ class BaseModel(object):
('numeric', 'float', get_pg_type(f)[1], '::'+get_pg_type(f)[1]), ('numeric', 'float', get_pg_type(f)[1], '::'+get_pg_type(f)[1]),
('float8', 'float', get_pg_type(f)[1], '::'+get_pg_type(f)[1]), ('float8', 'float', get_pg_type(f)[1], '::'+get_pg_type(f)[1]),
] ]
if f_pg_type == 'varchar' and f._type == 'char' and ((f.size is None and f_pg_size) or f_pg_size < f.size): if f_pg_type == 'varchar' and f._type == 'char' and f_pg_size and (f.size is None or f_pg_size < f.size):
try: try:
with cr.savepoint(): with cr.savepoint():
cr.execute('ALTER TABLE "%s" ALTER COLUMN "%s" TYPE %s' % (self._table, k, pg_varchar(f.size))) cr.execute('ALTER TABLE "%s" ALTER COLUMN "%s" TYPE %s' % (self._table, k, pg_varchar(f.size)))