mirror of
https://github.com/guohuadeng/app-odoo.git
synced 2025-02-23 04:11:36 +02:00
update common
This commit is contained in:
@@ -23,7 +23,10 @@
|
||||
# description:
|
||||
|
||||
from . import base
|
||||
# from . import fields
|
||||
# from . import validator
|
||||
# from . import ir_ui_view
|
||||
from . import fields
|
||||
from . import view_validation
|
||||
from . import ir_ui_view
|
||||
from . import ir_cron
|
||||
from . import res_users
|
||||
|
||||
|
||||
|
||||
@@ -2,34 +2,86 @@
|
||||
from odoo import models, fields, api, _
|
||||
from odoo.tools import DEFAULT_SERVER_DATE_FORMAT, DEFAULT_SERVER_DATETIME_FORMAT
|
||||
|
||||
import requests
|
||||
import base64
|
||||
from io import BytesIO
|
||||
|
||||
from datetime import date, datetime, time
|
||||
import pytz
|
||||
|
||||
import logging
|
||||
|
||||
_logger = logging.getLogger(__name__)
|
||||
|
||||
# 常规的排除的fields
|
||||
EXCLU_FIELDS = [
|
||||
'__last_update',
|
||||
'access_token',
|
||||
'access_url',
|
||||
'access_warning',
|
||||
'activity_date_deadline',
|
||||
'activity_exception_decoration',
|
||||
'activity_exception_icon',
|
||||
'activity_ids',
|
||||
'activity_state',
|
||||
'activity_summary',
|
||||
'activity_type_id',
|
||||
'activity_user_id',
|
||||
'display_name',
|
||||
'message_attachment_count',
|
||||
'message_channel_ids',
|
||||
'message_follower_ids',
|
||||
'message_has_error',
|
||||
'message_has_error_counter',
|
||||
'message_has_sms_error',
|
||||
'message_ids',
|
||||
'message_is_follower',
|
||||
'message_main_attachment_id',
|
||||
'message_needaction',
|
||||
'message_needaction_counter',
|
||||
'message_partner_ids',
|
||||
'message_unread',
|
||||
'message_unread_counter',
|
||||
'website_message_ids',
|
||||
'write_date',
|
||||
'write_uid',
|
||||
]
|
||||
|
||||
|
||||
class Base(models.AbstractModel):
|
||||
_inherit = 'base'
|
||||
|
||||
@api.model
|
||||
def _get_normal_fields(self):
|
||||
f_list = []
|
||||
for k, v in self._fields.items():
|
||||
if k not in EXCLU_FIELDS:
|
||||
f_list.append(k)
|
||||
return f_list
|
||||
|
||||
@api.model
|
||||
def _app_get_m2o_default(self, fieldname, domain=[]):
|
||||
if hasattr(self, fieldname) and self._fields[fieldname].type == 'many2one':
|
||||
if self._context.get(fieldname) or self._context.get('default_%s' % fieldname):
|
||||
return self._context.get(fieldname) or self._context.get('default_%s' % fieldname)
|
||||
else:
|
||||
rec = self.env[self._fields[fieldname].comodel_name].search(domain, limit=1)
|
||||
rec = self.env[self._fields[fieldname].comodel_name].sudo().search(domain, limit=1)
|
||||
return rec.id if rec else False
|
||||
return False
|
||||
|
||||
def _app_dt2local(self, value, return_format=DEFAULT_SERVER_DATETIME_FORMAT):
|
||||
"""
|
||||
将value中时间,按格式转为用户本地时间
|
||||
将value中时间,按格式转为用户本地时间.注意只处理in str为字符串类型,如果是时间类型直接用 datetime.now(tz)
|
||||
"""
|
||||
if not value:
|
||||
return value
|
||||
if isinstance(value, datetime):
|
||||
value = value.strftime(return_format)
|
||||
dt = datetime.strptime(value, return_format)
|
||||
pytz_timezone = pytz.timezone(self.env.user.tz or 'Etc/GMT-8')
|
||||
user_tz = pytz.timezone(self.env.user.tz or 'Etc/GMT-8')
|
||||
_logger.warning('============= user2 tz: %s' % user_tz)
|
||||
dt = dt.replace(tzinfo=pytz.timezone('UTC'))
|
||||
return dt.astimezone(pytz_timezone).strftime(return_format)
|
||||
return dt.astimezone(user_tz).strftime(return_format)
|
||||
|
||||
def _app_dt2utc(self, value, return_format=DEFAULT_SERVER_DATETIME_FORMAT):
|
||||
"""
|
||||
@@ -40,6 +92,17 @@ class Base(models.AbstractModel):
|
||||
if isinstance(value, datetime):
|
||||
value = value.strftime(return_format)
|
||||
dt = datetime.strptime(value, return_format)
|
||||
user_tz = pytz.timezone(self.env.user.tz or 'Etc/GMT+8')
|
||||
pytz_timezone = pytz.timezone('Etc/GMT+8')
|
||||
dt = dt.replace(tzinfo=pytz.timezone('UTC'))
|
||||
return dt.astimezone(user_tz).strftime(return_format)
|
||||
return dt.astimezone(pytz_timezone).strftime(return_format)
|
||||
|
||||
@api.model
|
||||
def get_image_from_url(self, url):
|
||||
if not url:
|
||||
return None
|
||||
try:
|
||||
response = requests.get(url) # 将这个图片保存在内存
|
||||
except Exception as e:
|
||||
return None
|
||||
# 返回这个图片的base64编码
|
||||
return base64.b64encode(BytesIO(response.content).read())
|
||||
|
||||
15
app_common/models/ir_cron.py
Normal file
15
app_common/models/ir_cron.py
Normal file
@@ -0,0 +1,15 @@
|
||||
# -*- coding: utf-8 -*-
|
||||
# Part of Odoo. See LICENSE file for full copyright and licensing details.
|
||||
import logging
|
||||
from odoo import api, fields, models, modules, tools, _
|
||||
|
||||
_logger = logging.getLogger(__name__)
|
||||
|
||||
class IrCron(models.Model):
|
||||
_inherit = "ir.cron"
|
||||
|
||||
trigger_user_id = fields.Many2one('res.users', string='Last Trigger User')
|
||||
|
||||
def method_direct_trigger(self):
|
||||
self.write({'trigger_user_id': self.env.user.id})
|
||||
return super(IrCron, self).method_direct_trigger()
|
||||
@@ -22,7 +22,7 @@ def app_relaxng(view_type):
|
||||
""" Return a validator for the given view type, or None. """
|
||||
if view_type not in _relaxng_cache:
|
||||
# tree, search 特殊
|
||||
if view_type in ['tree', 'search']:
|
||||
if view_type in ['tree', 'search', 'pivot']:
|
||||
_file = get_resource_path('app_common', 'rng', '%s_view.rng' % view_type)
|
||||
else:
|
||||
_file = get_resource_path('base', 'rng', '%s_view.rng' % view_type)
|
||||
@@ -31,7 +31,7 @@ def app_relaxng(view_type):
|
||||
relaxng_doc = etree.parse(frng)
|
||||
_relaxng_cache[view_type] = etree.RelaxNG(relaxng_doc)
|
||||
except Exception:
|
||||
_logger.exception('Failed to load RelaxNG XML schema for views validation')
|
||||
_logger.error('Failed to load RelaxNG XML schema for views validation')
|
||||
_relaxng_cache[view_type] = None
|
||||
return _relaxng_cache[view_type]
|
||||
|
||||
|
||||
10
app_common/models/res_users.py
Normal file
10
app_common/models/res_users.py
Normal file
@@ -0,0 +1,10 @@
|
||||
# -*- coding: utf-8 -*-
|
||||
# Part of Odoo. See LICENSE file for full copyright and licensing details.
|
||||
|
||||
from odoo import api, fields, models, tools, _
|
||||
|
||||
|
||||
class ResUsers(models.Model):
|
||||
_inherit = 'res.users'
|
||||
|
||||
login = fields.Char(index=True)
|
||||
63
app_common/models/view_validation.py
Normal file
63
app_common/models/view_validation.py
Normal file
@@ -0,0 +1,63 @@
|
||||
# -*- coding: utf-8 -*-
|
||||
|
||||
import ast
|
||||
from odoo.tools import view_validation
|
||||
from odoo.tools.view_validation import get_attrs_field_names as old_gafn
|
||||
from odoo.tools.view_validation import _get_attrs_symbols
|
||||
import logging
|
||||
|
||||
_logger = logging.getLogger(__name__)
|
||||
|
||||
ATTRS_WITH_FIELD_NAMES2 = {
|
||||
'context',
|
||||
'domain',
|
||||
'decoration-bf',
|
||||
'decoration-it',
|
||||
'decoration-danger',
|
||||
'decoration-info',
|
||||
'decoration-muted',
|
||||
'decoration-primary',
|
||||
'decoration-success',
|
||||
'decoration-warning',
|
||||
'decoration-black',
|
||||
'decoration-white',
|
||||
'bg-danger',
|
||||
'bg-info',
|
||||
'bg-muted',
|
||||
'bg-primary',
|
||||
'bg-success',
|
||||
'bg-warning',
|
||||
'bg-black',
|
||||
'bg-white',
|
||||
}
|
||||
|
||||
def app_get_attrs_field_names(env, arch, model, editable):
|
||||
symbols = _get_attrs_symbols() | {None}
|
||||
result = []
|
||||
|
||||
def get_name(node):
|
||||
""" return the name from an AST node, or None """
|
||||
if isinstance(node, ast.Name):
|
||||
return node.id
|
||||
|
||||
def process_expr(expr, get, key, val):
|
||||
""" parse `expr` and collect triples """
|
||||
for node in ast.walk(ast.parse(expr.strip(), mode='eval')):
|
||||
name = get(node)
|
||||
if name not in symbols:
|
||||
result.append((name, key, val))
|
||||
|
||||
def add_bg(node, model, editable, get=get_name):
|
||||
for key, val in node.items():
|
||||
if not val:
|
||||
continue
|
||||
if key in ATTRS_WITH_FIELD_NAMES2:
|
||||
process_expr(val, get, key, val)
|
||||
|
||||
res = old_gafn(env, arch, model, editable)
|
||||
add_bg(arch, model, editable)
|
||||
res += result
|
||||
return res
|
||||
|
||||
# 使用猴子补丁方式更新
|
||||
view_validation.get_attrs_field_names = app_get_attrs_field_names
|
||||
Reference in New Issue
Block a user