Compare commits

..

2 Commits

Author SHA1 Message Date
胡尧
307e860fe0 增加提示模块 2025-06-20 15:16:43 +08:00
胡尧
bd27f288f7 增加提示模块 2025-06-20 15:16:25 +08:00
14 changed files with 282 additions and 484 deletions

View File

@@ -190,7 +190,7 @@ def _create(self, data_list):
# 如果该用户组被限制创建或更新操作 # 如果该用户组被限制创建或更新操作
if rec['is_create_or_update']: if rec['is_create_or_update']:
raise UserError( raise UserError(
_("您没有执行此操作的权限。请联系管理员")) _("您没有执行此操作%s的权限。请联系管理员" % group_xml_id))
else: else:
# 如果 'access.right' 模型不存在,可以在这里定义备选逻辑 # 如果 'access.right' 模型不存在,可以在这里定义备选逻辑
# 例如,记录日志、发送通知或者简单地跳过这部分逻辑 # 例如,记录日志、发送通知或者简单地跳过这部分逻辑

View File

@@ -10,7 +10,7 @@
""", """,
'category': 'sf', 'category': 'sf',
'website': 'https://www.sf.jikimo.com', 'website': 'https://www.sf.jikimo.com',
'depends': ['sf_plan','jikimo_printing'], 'depends': ['sf_plan'],
'data': [ 'data': [
'security/ir.model.access.csv', 'security/ir.model.access.csv',
'views/demand_plan.xml', 'views/demand_plan.xml',

View File

@@ -403,30 +403,29 @@ class SfProductionDemandPlan(models.Model):
outsourcing_purchase_request.extend(pr_ids.ids) outsourcing_purchase_request.extend(pr_ids.ids)
if record.supply_method == 'outsourcing' and not record.sale_order_line_id.is_incoming_material: if record.supply_method == 'outsourcing' and not record.sale_order_line_id.is_incoming_material:
bom_line_ids = record.product_id.bom_ids.bom_line_ids bom_line_ids = record.product_id.bom_ids.bom_line_ids
if bom_line_ids: # BOM_数量
# BOM_数量 total_product_qty = sum(line.product_qty for line in bom_line_ids)
total_product_qty = sum(line.product_qty for line in bom_line_ids) bom_product_ids = bom_line_ids.mapped('product_id')
bom_product_ids = bom_line_ids.mapped('product_id') product_purchase_orders = self.env['purchase.order'].sudo().search([
product_purchase_orders = self.env['purchase.order'].sudo().search([ ('state', 'in', ('purchase', 'done')),
('state', 'in', ('purchase', 'done')), ('order_line.product_id', 'in', bom_product_ids.ids)
('order_line.product_id', 'in', bom_product_ids.ids) ])
]) # 购订单_数量
# 购订单_数量 total_outsourcing_purchase_quantity = sum(
total_outsourcing_purchase_quantity = sum( sum(
sum( order.order_line.filtered(
order.order_line.filtered( lambda line: line.product_id in bom_product_ids
lambda line: line.product_id in bom_product_ids ).mapped('product_qty')
).mapped('product_qty')
)
for order in product_purchase_orders
) )
quantity = total_outsourcing_purchase_quantity / total_product_qty for order in product_purchase_orders
if float_compare(quantity, record.product_uom_qty, )
precision_rounding=record.product_id.uom_id.rounding) == -1: quantity = total_outsourcing_purchase_quantity / total_product_qty
purchase_request = self.env['purchase.request'].sudo().search( if float_compare(quantity, record.product_uom_qty,
[('line_ids.product_id', 'in', bom_product_ids.ids), precision_rounding=record.product_id.uom_id.rounding) == -1:
('line_ids.purchase_state', 'not in', ('purchase', 'done')), ('state', '!=', 'done')]) purchase_request = self.env['purchase.request'].sudo().search(
outsourcing_purchase_request.extend(purchase_request.ids) [('line_ids.product_id', 'in', bom_product_ids.ids),
('line_ids.purchase_state', 'not in', ('purchase', 'done')), ('state', '!=', 'done')])
outsourcing_purchase_request.extend(purchase_request.ids)
record.outsourcing_purchase_request = json.dumps(outsourcing_purchase_request) record.outsourcing_purchase_request = json.dumps(outsourcing_purchase_request)
if outsourcing_purchase_request: if outsourcing_purchase_request:
record.hide_action_purchase_orders = True record.hide_action_purchase_orders = True

View File

@@ -4,12 +4,12 @@
<field name="name">sf.demand.plan.print.wizard.tree</field> <field name="name">sf.demand.plan.print.wizard.tree</field>
<field name="model">sf.demand.plan.print.wizard</field> <field name="model">sf.demand.plan.print.wizard</field>
<field name="arch" type="xml"> <field name="arch" type="xml">
<tree string="打印" class="print_demand" js_class="print_demand"> <tree string="打印" class="print_demand" js_class="print_demand" >
<field name="model_id"/> <field name="model_id"/>
<field name="filename_url"/> <field name="filename_url"/>
<field name="type"/> <field name="type"/>
<field name="machining_drawings" attrs="{'column_invisible': True }"/> <field name="machining_drawings" attrs="{'column_invisible': True }"/>
<field name="cnc_worksheet" attrs="{'column_invisible': True }"/> <field name="cnc_worksheet" attrs="{'column_invisible': True }" />
<field name="status"/> <field name="status"/>
</tree> </tree>
</field> </field>

View File

@@ -437,7 +437,7 @@ class Sf_Dashboard_Connect(http.Controller):
('state', 'in', ['ready', 'progress', 'done']) ('state', 'in', ['ready', 'progress', 'done'])
]) ])
plan_data_total_counts = sum(plan_data_total.mapped('qty_production')) plan_data_total_counts = sum(plan_data_total.mapped('qty_produced'))
# # 工单完成量 # # 工单完成量
# plan_data_finish_counts = plan_obj.search_count( # plan_data_finish_counts = plan_obj.search_count(
@@ -489,7 +489,7 @@ class Sf_Dashboard_Connect(http.Controller):
# 工单返工数量 # 工单返工数量
plan_data_rework = work_order_obj.search(work_order_domain + [ plan_data_rework = work_order_obj.search(plan_domain + [
('state', 'in', ['rework']) ('state', 'in', ['rework'])
]) ])
@@ -601,11 +601,8 @@ class Sf_Dashboard_Connect(http.Controller):
line_list = ast.literal_eval(kw['line_list']) line_list = ast.literal_eval(kw['line_list'])
begin_time_str = kw['begin_time'].strip('"') begin_time_str = kw['begin_time'].strip('"')
end_time_str = kw['end_time'].strip('"') end_time_str = kw['end_time'].strip('"')
# 将时间减去8小时UTC+8转UTC begin_time = datetime.strptime(begin_time_str, '%Y-%m-%d %H:%M:%S')
begin_time = (datetime.strptime(begin_time_str, '%Y-%m-%d %H:%M:%S') - timedelta(hours=8)) end_time = datetime.strptime(end_time_str, '%Y-%m-%d %H:%M:%S')
end_time = (datetime.strptime(end_time_str, '%Y-%m-%d %H:%M:%S') - timedelta(hours=8))
# begin_time = datetime.strptime(begin_time_str, '%Y-%m-%d %H:%M:%S')
# end_time = datetime.strptime(end_time_str, '%Y-%m-%d %H:%M:%S')
# print('line_list: %s' % line_list) # print('line_list: %s' % line_list)
print('kw', kw) print('kw', kw)
time_unit = kw.get('time_unit', 'day').strip('"') # 默认单位为天 time_unit = kw.get('time_unit', 'day').strip('"') # 默认单位为天
@@ -640,15 +637,6 @@ class Sf_Dashboard_Connect(http.Controller):
if time_unit == 'hour': if time_unit == 'hour':
# 计划量目前只能从mail.message中筛选出
plan_order_messages = request.env['mail.message'].sudo().search([
('model', '=', 'mrp.workorder'),
('create_date', '>=', begin_time.strftime('%Y-%m-%d %H:%M:%S')),
('create_date', '<=', end_time.strftime('%Y-%m-%d %H:%M:%S')),
('tracking_value_ids.field_desc', '=', '状态'),
('tracking_value_ids.new_value_char', '=', '就绪')
])
for line in line_list: for line in line_list:
date_field_name = 'date_finished' # 替换为你模型中的实际字段名 date_field_name = 'date_finished' # 替换为你模型中的实际字段名
order_counts = [] order_counts = []
@@ -690,10 +678,19 @@ class Sf_Dashboard_Connect(http.Controller):
) )
# 使用小时和分钟作为键,确保每个小时的数据有独立的键 # 使用小时和分钟作为键,确保每个小时的数据有独立的键
key = (start_time + timedelta(hours=8)).strftime('%H:%M:%S') # 只取小时:分钟:秒作为键 key = start_time.strftime('%H:%M:%S') # 只取小时:分钟:秒作为键
# time_count_dict[key] = len(orders) # time_count_dict[key] = len(orders)
time_count_dict[key] = sum(interval_orders.mapped('qty_produced')) time_count_dict[key] = sum(interval_orders.mapped('qty_produced'))
# 计划量目前只能从mail.message中筛选出
plan_order_messages = request.env['mail.message'].sudo().search([
('model', '=', 'mrp.workorder'),
('create_date', '>=', begin_time.strftime('%Y-%m-%d %H:%M:%S')),
('create_date', '<=', end_time.strftime('%Y-%m-%d %H:%M:%S')),
('tracking_value_ids.field_desc', '=', '状态'),
('tracking_value_ids.new_value_char', '=', '就绪')
])
for time_interval in time_intervals: for time_interval in time_intervals:
start_time, end_time = time_interval start_time, end_time = time_interval
@@ -718,9 +715,9 @@ class Sf_Dashboard_Connect(http.Controller):
interval_orders = interval_orders.filtered(lambda o: o.routing_type == 'CNC加工' and o.production_line_id.name == line) interval_orders = interval_orders.filtered(lambda o: o.routing_type == 'CNC加工' and o.production_line_id.name == line)
# 使用小时和分钟作为键,确保每个小时的数据有独立的键 # 使用小时和分钟作为键,确保每个小时的数据有独立的键
key = (start_time + timedelta(hours=8)).strftime('%H:%M:%S') # 只取小时:分钟:秒作为键 key = start_time.strftime('%H:%M:%S') # 只取小时:分钟:秒作为键
# time_count_dict[key] = len(orders) # time_count_dict[key] = len(orders)
plan_count_dict[key] = sum(interval_orders.mapped('qty_production')) plan_count_dict[key] = sum(interval_orders.mapped('qty_produced'))
# order_counts.append() # order_counts.append()
res['data'][line] = { res['data'][line] = {
@@ -860,6 +857,7 @@ class Sf_Dashboard_Connect(http.Controller):
:param kw: :param kw:
:return: :return:
""" """
request.env['stock.warehouse'].browse(request.env.company.id).pbm_loc_id
# res = {'status': 1, 'message': '成功', 'not_done_data': [], 'done_data': []} # res = {'status': 1, 'message': '成功', 'not_done_data': [], 'done_data': []}
res = {'status': 1, 'message': '成功', 'data': {}} res = {'status': 1, 'message': '成功', 'data': {}}
plan_obj = request.env['sf.production.plan'].sudo() plan_obj = request.env['sf.production.plan'].sudo()
@@ -876,18 +874,6 @@ class Sf_Dashboard_Connect(http.Controller):
not_done_index = 1 not_done_index = 1
done_index = 1 done_index = 1
# 获取当前时间并计算24小时前的时间
current_time = datetime.now()
time_48_hours_ago = current_time - timedelta(hours=48)
# 计划量目前只能从mail.message中筛选出
plan_order_messages = request.env['mail.message'].sudo().search([
('model', '=', 'mrp.workorder'),
('create_date', '>=', time_48_hours_ago.strftime('%Y-%m-%d %H:%M:%S')),
('tracking_value_ids.field_desc', '=', '状态'),
('tracking_value_ids.new_value_char', 'in', ['就绪', '生产中'])
])
for line in line_list: for line in line_list:
if line == '业绩总览': if line == '业绩总览':
@@ -904,27 +890,19 @@ class Sf_Dashboard_Connect(http.Controller):
# [('production_line_id.name', '=', line), ('state', 'not in', ['finished']), # [('production_line_id.name', '=', line), ('state', 'not in', ['finished']),
# ('production_id.state', 'not in', ['cancel', 'done']), ('active', '=', True) # ('production_id.state', 'not in', ['cancel', 'done']), ('active', '=', True)
# ]) # ])
# not_done_orders = work_order_obj.search(work_order_domain + not_done_orders = work_order_obj.search(work_order_domain +
# [('state', 'in', ['ready', 'progress'])], order='id asc' [('state', 'in', ['ready', 'progress'])], order='id asc'
# ) )
not_done_orders = request.env['mrp.workorder'].sudo().browse(plan_order_messages.mapped('res_id'))
if line == '业绩总览':
not_done_orders = not_done_orders.filtered(lambda o: o.routing_type in ['人工线下加工', 'CNC加工'])
elif line == '人工线下加工中心':
not_done_orders = not_done_orders.filtered(lambda o: o.routing_type == '人工线下加工')
else:
not_done_orders = not_done_orders.filtered(lambda o: o.routing_type == 'CNC加工' and o.production_line_id.name == line)
# 完成订单 # 完成订单
# 获取当前时间并计算24小时前的时间 # 获取当前时间并计算24小时前的时间
# current_time = datetime.now() current_time = datetime.now()
# time_24_hours_ago = current_time - timedelta(hours=24) time_24_hours_ago = current_time - timedelta(hours=24)
finish_orders = work_order_obj.search(work_order_domain + [ finish_orders = work_order_obj.search(work_order_domain + [
('state', 'in', ['finished']), ('state', 'in', ['finished']),
('production_id.state', 'not in', ['cancel']), ('production_id.state', 'not in', ['cancel']),
('date_finished', '>=', time_48_hours_ago) ('date_finished', '>=', time_24_hours_ago)
], order='id asc') ], order='id asc')
# print(finish_orders) # print(finish_orders)

View File

@@ -18,5 +18,4 @@ from . import quick_easy_order
from . import purchase_order from . import purchase_order
from . import quality_check from . import quality_check
from . import purchase_request_line from . import purchase_request_line
from . import bom
# from . import stock_warehouse_orderpoint # from . import stock_warehouse_orderpoint

View File

@@ -1,16 +0,0 @@
from odoo import models
from odoo.osv.expression import AND
class MrpBom(models.Model):
_inherit = 'mrp.bom'
def _bom_subcontract_find(self, product, picking_type=None, company_id=False, bom_type='subcontract', subcontractor=False):
domain = self._bom_find_domain(product, picking_type=picking_type, company_id=company_id, bom_type=bom_type)
if self.env.context.get('stock_picking') == 'outsourcing':
return self.search(domain, order='sequence, product_id, id', limit=1)
if subcontractor:
domain = AND([domain, [('subcontractor_ids', 'parent_of', subcontractor.ids)]])
return self.search(domain, order='sequence, product_id, id', limit=1)
else:
return self.env['mrp.bom']

View File

@@ -1709,77 +1709,7 @@ class MrpProduction(models.Model):
vals['procurement_group_id'] = product_group_id[product_id.id] vals['procurement_group_id'] = product_group_id[product_id.id]
else: else:
vals['procurement_group_id'] = is_custemer_group_id[key] vals['procurement_group_id'] = is_custemer_group_id[key]
return super(MrpProduction, self).create(vals_list)
productions = super(MrpProduction, self).create(vals_list)
# 查询成品工序排序(自动化产线加工),供后续使用
product_model_type_routing_sorts = self.env['sf.product.model.type.routing.sort'].search([], order='sequence asc')
# 查询成品工序排序(人工线下加工),供后续使用
manual_product_model_type_routing_sorts = self.env['sf.manual.product.model.type.routing.sort'].search([], order='sequence asc')
# 查询坯料工序排序,供后续使用
embryo_model_type_routing_sorts = self.env['sf.embryo.model.type.routing.sort'].search([], order='sequence asc')
for production in productions:
# 生成序列号
production.action_generate_serial()
# 创建工序模板
technology_design_values = []
i = 0
if production.product_id.categ_id.type == '成品':
# 根据加工面板的面数及成品工序模板生成工序设计
if production.production_type == '自动化产线加工':
product_routing_workcenter = product_model_type_routing_sorts.filtered(
lambda s: s.product_model_type_id.id == production.product_id.product_model_type_id.id
)
else:
product_routing_workcenter = manual_product_model_type_routing_sorts.filtered(
lambda s: s.manual_product_model_type_id.id == production.product_id.product_model_type_id.id
)
if production.production_type == '自动化产线加工':
for k in (production.product_id.model_processing_panel.split(',')):
for route in product_routing_workcenter:
i += 1
technology_design_values.append(
self.env['sf.technology.design'].json_technology_design_str(k, route, i, False)
)
elif production.production_type == '人工线下加工':
for route in product_routing_workcenter:
i += 1
technology_design_values.append(
self.env['sf.technology.design'].json_technology_design_str('ZM', route, i, False)
)
else:
for route in product_routing_workcenter:
i += 1
technology_design_values.append(
self.env['sf.technology.design'].json_technology_design_str(False, route, i, False)
)
elif production.product_id.categ_id.type == '坯料':
embryo_routing_workcenter = embryo_model_type_routing_sorts.filtered(
lambda s: s.embryo_model_type_id.id == production.product_id.embryo_model_type_id.id
)
for route_embryo in embryo_routing_workcenter:
i += 1
technology_design_values.append(
self.env['sf.technology.design'].json_technology_design_str(False, route_embryo, i, False)
)
# 处理表面工艺
for item in production.product_id.product_model_type_id.surface_technics_routing_tmpl_ids:
if item.route_workcenter_id.surface_technics_id.id:
for process_param in production.product_id.model_process_parameters_ids:
if item.route_workcenter_id.surface_technics_id == process_param.process_id:
technology_design_values.append(
self.env['sf.technology.design'].json_technology_design_str(
False,
item.route_workcenter_id,
i,
process_param
)
)
production.technology_design_ids = technology_design_values
# 设置制造订单状态为待工艺确认
productions.write({'state': 'technology_to_confirmed'})
return productions
@api.depends('procurement_group_id.stock_move_ids.created_purchase_line_id.order_id', @api.depends('procurement_group_id.stock_move_ids.created_purchase_line_id.order_id',
'procurement_group_id.stock_move_ids.move_orig_ids.purchase_line_id.order_id') 'procurement_group_id.stock_move_ids.move_orig_ids.purchase_line_id.order_id')

View File

@@ -95,36 +95,35 @@ class StockRule(models.Model):
precision_rounding=proc[ precision_rounding=proc[
0].product_uom.rounding) > 0) 0].product_uom.rounding) > 0)
list2 = [] list2 = []
for procurement, rule in procurements: for item in procurements:
num = int(procurement.product_qty) num = int(item[0].product_qty)
warehouse_id = rule.warehouse_id product = self.env['product.product'].search(
if not warehouse_id: [("id", '=', item[0].product_id.id)])
warehouse_id = rule.location_dest_id.warehouse_id product_tmpl = self.env['product.template'].search(
manu_rule = rule.route_id.rule_ids.filtered(lambda r: r.action == 'manufacture' and r.warehouse_id == warehouse_id) ["&", ("id", '=', product.product_tmpl_id.id), ('single_manufacturing', "!=", False)])
if product_tmpl:
if procurement.product_id.product_tmpl_id.single_manufacturing and manu_rule:
if num > 1: if num > 1:
for no in range(1, num + 1): for no in range(1, num + 1):
Procurement = namedtuple('Procurement', ['product_id', 'product_qty', Procurement = namedtuple('Procurement', ['product_id', 'product_qty',
'product_uom', 'location_id', 'name', 'origin', 'product_uom', 'location_id', 'name', 'origin',
'company_id', 'company_id',
'values']) 'values'])
s = Procurement(product_id=procurement.product_id, product_qty=1.0, product_uom=procurement.product_uom, s = Procurement(product_id=item[0].product_id, product_qty=1.0, product_uom=item[0].product_uom,
location_id=procurement.location_id, location_id=item[0].location_id,
name=procurement.name, name=item[0].name,
origin=procurement.origin, origin=item[0].origin,
company_id=procurement.company_id, company_id=item[0].company_id,
values=procurement.values, values=item[0].values,
) )
# item1 = list(item) item1 = list(item)
# item1[0] = s item1[0] = s
list2.append((s, rule)) list2.append(tuple(item1))
else: else:
list2.append((procurement, rule)) list2.append(item)
else: else:
list2.append((procurement, rule)) list2.append(item)
for procurement, rule in list2: for procurement, rule in list2:
procure_method = rule.procure_method procure_method = rule.procure_method
@@ -163,203 +162,215 @@ class StockRule(models.Model):
[('res_id', '=', res_id), ('res_field', '=', res_field)], limit=1) [('res_id', '=', res_id), ('res_field', '=', res_field)], limit=1)
attachment_info.write({'name': name}) attachment_info.write({'name': name})
# @api.model @api.model
# def _run_manufacture(self, procurements): def _run_manufacture(self, procurements):
# productions_values_by_company = defaultdict(list) productions_values_by_company = defaultdict(list)
# errors = [] errors = []
# for procurement, rule in procurements: for procurement, rule in procurements:
# if float_compare(procurement.product_qty, 0, precision_rounding=procurement.product_uom.rounding) <= 0: if float_compare(procurement.product_qty, 0, precision_rounding=procurement.product_uom.rounding) <= 0:
# # If procurement contains negative quantity, don't create a MO that would be for a negative value. # If procurement contains negative quantity, don't create a MO that would be for a negative value.
# continue continue
# bom = rule._get_matching_bom(procurement.product_id, procurement.company_id, procurement.values) bom = rule._get_matching_bom(procurement.product_id, procurement.company_id, procurement.values)
# productions_values_by_company[procurement.company_id.id].append(rule._prepare_mo_vals(*procurement, bom)) productions_values_by_company[procurement.company_id.id].append(rule._prepare_mo_vals(*procurement, bom))
# if errors: if errors:
# raise ProcurementException(errors) raise ProcurementException(errors)
# for company_id, productions_values in productions_values_by_company.items(): for company_id, productions_values in productions_values_by_company.items():
# # create the MO as SUPERUSER because the current user may not have the rights to do it # create the MO as SUPERUSER because the current user may not have the rights to do it
# # (mto product launched by a sale for example) # (mto product launched by a sale for example)
# '''创建制造订单''' '''创建制造订单'''
# productions = self.env['mrp.production'].with_user(SUPERUSER_ID).sudo().with_company(company_id).create( productions = self.env['mrp.production'].with_user(SUPERUSER_ID).sudo().with_company(company_id).create(
# productions_values) productions_values)
# 将这一批制造订单的采购组根据成品设置为不同的采购组
# product_group_id = {}
# for index, production in enumerate(productions):
# if production.product_id.id not in product_group_id.keys():
# product_group_id[production.product_id.id] = production.procurement_group_id.id
# else:
# productions_values[index].update({'name': production.name})
# procurement_group_vals = production._prepare_procurement_group_vals(productions_values[index])
# production.procurement_group_id = self.env["procurement.group"].create(procurement_group_vals).id
# ''' # self.env['stock.move'].sudo().create(productions._get_moves_raw_values())
# 创建工单 # self.env['stock.move'].sudo().create(productions._get_moves_finished_values())
# '''
# # productions._create_workorder()
# #
# # self.env['stock.move'].sudo().create(productions._get_moves_finished_values())
# productions.filtered(lambda p: (not p.orderpoint_id and p.move_raw_ids) or \
# (
# p.move_dest_ids.procure_method != 'make_to_order' and not
# p.move_raw_ids and not p.workorder_ids)).action_confirm()
# # 处理 根据制造订单生成的采购单坯料入库时到原材料库,手动将原材料位置该为坯料存货区
# for production in productions:
# if production.picking_ids:
# product_type_id = production.picking_ids[0].move_ids[0].product_id.categ_id
# if product_type_id.name == '坯料':
# location_id = self.env['stock.location'].search([('name', '=', '坯料存货区')])
# if not location_id:
# logging.info(f'没有搜索到【坯料存货区】: {location_id}')
# break
# for picking_id in production.picking_ids:
# if picking_id.picking_type_id.name == '内部调拨':
# if picking_id.location_dest_id.product_type != product_type_id:
# picking_id.location_dest_id = location_id.id
# elif picking_id.picking_type_id.name == '生产发料':
# if picking_id.location_id.product_type != product_type_id:
# picking_id.location_id = location_id.id
# for production in productions: '''
# ''' 创建工单
# 创建制造订单时生成序列号 '''
# ''' # productions._create_workorder()
# # production.action_generate_serial() #
# origin_production = production.move_dest_ids and production.move_dest_ids[ # self.env['stock.move'].sudo().create(productions._get_moves_finished_values())
# 0].raw_material_production_id or False productions.filtered(lambda p: (not p.orderpoint_id and p.move_raw_ids) or \
# orderpoint = production.orderpoint_id (
# if orderpoint and orderpoint.create_uid.id == SUPERUSER_ID and orderpoint.trigger == 'manual': p.move_dest_ids.procure_method != 'make_to_order' and not
# production.message_post( p.move_raw_ids and not p.workorder_ids)).action_confirm()
# body=_('This production order has been created from Replenishment Report.'), # 处理 根据制造订单生成的采购单坯料入库时到原材料库,手动将原材料位置该为坯料存货区
# message_type='comment', for production in productions:
# subtype_xmlid='mail.mt_note') if production.picking_ids:
# elif orderpoint: product_type_id = production.picking_ids[0].move_ids[0].product_id.categ_id
# production.message_post_with_view( if product_type_id.name == '坯料':
# 'mail.message_origin_link', location_id = self.env['stock.location'].search([('name', '=', '坯料存货区')])
# values={'self': production, 'origin': orderpoint}, if not location_id:
# subtype_id=self.env.ref('mail.mt_note').id) logging.info(f'没有搜索到【坯料存货区】: {location_id}')
# elif origin_production: break
# production.message_post_with_view( for picking_id in production.picking_ids:
# 'mail.message_origin_link', if picking_id.picking_type_id.name == '内部调拨':
# values={'self': production, 'origin': origin_production}, if picking_id.location_dest_id.product_type != product_type_id:
# subtype_id=self.env.ref('mail.mt_note').id) picking_id.location_dest_id = location_id.id
elif picking_id.picking_type_id.name == '生产发料':
if picking_id.location_id.product_type != product_type_id:
picking_id.location_id = location_id.id
# ''' for production in productions:
# 创建生产计划 '''
# ''' 创建制造订单时生成序列号
# # 工单耗时 '''
# # workorder_duration = 0 production.action_generate_serial()
# # for workorder in production.workorder_ids: origin_production = production.move_dest_ids and production.move_dest_ids[
# # workorder_duration += workorder.duration_expected 0].raw_material_production_id or False
orderpoint = production.orderpoint_id
if orderpoint and orderpoint.create_uid.id == SUPERUSER_ID and orderpoint.trigger == 'manual':
production.message_post(
body=_('This production order has been created from Replenishment Report.'),
message_type='comment',
subtype_xmlid='mail.mt_note')
elif orderpoint:
production.message_post_with_view(
'mail.message_origin_link',
values={'self': production, 'origin': orderpoint},
subtype_id=self.env.ref('mail.mt_note').id)
elif origin_production:
production.message_post_with_view(
'mail.message_origin_link',
values={'self': production, 'origin': origin_production},
subtype_id=self.env.ref('mail.mt_note').id)
# # sale_order = self.env['sale.order'].sudo().search([('name', '=', production.origin)]) '''
# # # 如果订单为空,则获取来源制造订单的销售单 创建生产计划
# # if not sale_order: '''
# # mrp_production = self.env['mrp.production'].sudo().search([('name', '=', production.origin)], # 工单耗时
# # limit=1) workorder_duration = 0
# # if mrp_production: for workorder in production.workorder_ids:
# # sale_order = self.env['sale.order'].sudo().search([('name', '=', mrp_production.origin)]) workorder_duration += workorder.duration_expected
# # else:
# # mrp_production = production sale_order = self.env['sale.order'].sudo().search([('name', '=', production.origin)])
# # # if sale_order: # 如果订单为空,则获取来源制造订单的销售单
# # # sale_order.write({'schedule_status': 'to schedule'}) if not sale_order:
# # self.env['sf.production.plan'].sudo().with_company(company_id).create({ mrp_production = self.env['mrp.production'].sudo().search([('name', '=', production.origin)],
# # 'name': production.name, limit=1)
# # 'order_deadline': sale_order.deadline_of_delivery, if mrp_production:
# # 'production_id': production.id, sale_order = self.env['sale.order'].sudo().search([('name', '=', mrp_production.origin)])
# # 'date_planned_start': production.date_planned_start, else:
# # 'origin': mrp_production.origin, mrp_production = production
# # 'product_qty': production.product_qty, # if sale_order:
# # 'product_id': production.product_id.id, # sale_order.write({'schedule_status': 'to schedule'})
# # 'state': 'draft', self.env['sf.production.plan'].sudo().with_company(company_id).create({
# # }) 'name': production.name,
# all_production = productions 'order_deadline': sale_order.deadline_of_delivery,
# grouped_product_ids = {k: list(g) for k, g in groupby(all_production, key=lambda x: x.product_id.id)} 'production_id': production.id,
# # 初始化一个字典来存储每个product_id对应的生产订单名称列表 'date_planned_start': production.date_planned_start,
# product_id_to_production_names = {} 'origin': mrp_production.origin,
# # 对于每个product_id获取其所有生产订单的名称 'product_qty': production.product_qty,
# for product_id, all_production in grouped_product_ids.items(): 'product_id': production.product_id.id,
# # 为同一个product_id创建一个生产订单名称列表 'state': 'draft',
# product_id_to_production_names[product_id] = [production.name for production in all_production] })
# for production_item in productions: all_production = productions
# technology_design_values = [] grouped_product_ids = {k: list(g) for k, g in groupby(all_production, key=lambda x: x.product_id.id)}
# # production_programming = self.env['mrp.production'].search( # 初始化一个字典来存储每个product_id对应的生产订单名称列表
# # [('product_id.id', '=', production_item.product_id.id), product_id_to_production_names = {}
# # ('origin', '=', production_item.origin)], # 对于每个product_id获取其所有生产订单的名称
# # limit=1, order='id asc') for product_id, all_production in grouped_product_ids.items():
# # if production_item.product_id.id in product_id_to_production_names: # 为同一个product_id创建一个生产订单名称列表
# # # 同一个产品多个制造订单对应一个编程单和模型库 product_id_to_production_names[product_id] = [production.name for production in all_production]
# # # 只调用一次fetchCNC并将所有生产订单的名称作为字符串传递 for production_item in productions:
# # if not production_item.programming_no and production_item.production_type in ['自动化产线加工', technology_design_values = []
# # '人工线下加工']: production_programming = self.env['mrp.production'].search(
# # if not production_programming.programming_no: [('product_id.id', '=', production_item.product_id.id),
# # production_item.fetchCNC( ('origin', '=', production_item.origin)],
# # ', '.join(product_id_to_production_names[production_item.product_id.id])) limit=1, order='id asc')
# # else: if production_item.product_id.id in product_id_to_production_names:
# # production_item.write({'programming_no': production_programming.programming_no, # 同一个产品多个制造订单对应一个编程单和模型库
# # 'programming_state': '编程中'}) # 只调用一次fetchCNC并将所有生产订单的名称作为字符串传递
# i = 0 if not production_item.programming_no and production_item.production_type in ['自动化产线加工',
# if production_item.product_id.categ_id.type == '成品': '人工线下加工']:
# # 根据加工面板的面数及成品工序模板生成工序设计 if not production_programming.programming_no:
# if production_item.production_type == '自动化产线加工': production_item.fetchCNC(
# model = 'sf.product.model.type.routing.sort' ', '.join(product_id_to_production_names[production_item.product_id.id]))
# domain = [ else:
# ('product_model_type_id', '=', production_item.product_id.product_model_type_id.id)] production_item.write({'programming_no': production_programming.programming_no,
# else: 'programming_state': '编程中'})
# model = 'sf.manual.product.model.type.routing.sort' i = 0
# domain = [('manual_product_model_type_id', '=', if production_item.product_id.categ_id.type == '成品':
# production_item.product_id.product_model_type_id.id)] # 根据加工面板的面数及成品工序模板生成工序设计
# product_routing_workcenter = self.env[model].search(domain, order='sequence asc') if production_item.production_type == '自动化产线加工':
# if production_item.production_type == '自动化产线加工': model = 'sf.product.model.type.routing.sort'
# for k in (production_item.product_id.model_processing_panel.split(',')): domain = [
# for route in product_routing_workcenter: ('product_model_type_id', '=', production_item.product_id.product_model_type_id.id)]
# i += 1 else:
# technology_design_values.append( model = 'sf.manual.product.model.type.routing.sort'
# self.env['sf.technology.design'].json_technology_design_str(k, route, i, False)) domain = [('manual_product_model_type_id', '=',
# elif production_item.production_type == '人工线下加工': production_item.product_id.product_model_type_id.id)]
# for route in product_routing_workcenter: product_routing_workcenter = self.env[model].search(domain, order='sequence asc')
# i += 1 if production_item.production_type == '自动化产线加工':
# technology_design_values.append( for k in (production_item.product_id.model_processing_panel.split(',')):
# self.env['sf.technology.design'].json_technology_design_str('ZM', route, i, False)) for route in product_routing_workcenter:
# else: i += 1
# for route in product_routing_workcenter: technology_design_values.append(
# i += 1 self.env['sf.technology.design'].json_technology_design_str(k, route, i, False))
# technology_design_values.append( elif production_item.production_type == '人工线下加工':
# self.env['sf.technology.design'].json_technology_design_str(False, route, i, False)) for route in product_routing_workcenter:
# elif production_item.product_id.categ_id.type == '坯料': i += 1
# embryo_routing_workcenter = self.env['sf.embryo.model.type.routing.sort'].search( technology_design_values.append(
# [('embryo_model_type_id', '=', production_item.product_id.embryo_model_type_id.id)], self.env['sf.technology.design'].json_technology_design_str('ZM', route, i, False))
# order='sequence asc' else:
# ) for route in product_routing_workcenter:
# for route_embryo in embryo_routing_workcenter: i += 1
# i += 1 technology_design_values.append(
# technology_design_values.append( self.env['sf.technology.design'].json_technology_design_str(False, route, i, False))
# self.env['sf.technology.design'].json_technology_design_str(False, route_embryo, i, elif production_item.product_id.categ_id.type == '坯料':
# False)) embryo_routing_workcenter = self.env['sf.embryo.model.type.routing.sort'].search(
# surface_technics_arr = [] [('embryo_model_type_id', '=', production_item.product_id.embryo_model_type_id.id)],
# route_workcenter_arr = [] order='sequence asc'
# for item in production_item.product_id.product_model_type_id.surface_technics_routing_tmpl_ids: )
# if item.route_workcenter_id.surface_technics_id.id: for route_embryo in embryo_routing_workcenter:
# for process_param in production_item.product_id.model_process_parameters_ids: i += 1
# if item.route_workcenter_id.surface_technics_id == process_param.process_id: technology_design_values.append(
# surface_technics_arr.append( self.env['sf.technology.design'].json_technology_design_str(False, route_embryo, i,
# item.route_workcenter_id.surface_technics_id.id) False))
# route_workcenter_arr.append(item.route_workcenter_id.id) surface_technics_arr = []
# if surface_technics_arr: route_workcenter_arr = []
# production_process = self.env['sf.production.process'].search( for item in production_item.product_id.product_model_type_id.surface_technics_routing_tmpl_ids:
# [('id', 'in', surface_technics_arr)], if item.route_workcenter_id.surface_technics_id.id:
# order='sequence asc' for process_param in production_item.product_id.model_process_parameters_ids:
# ) if item.route_workcenter_id.surface_technics_id == process_param.process_id:
# for p in production_process: surface_technics_arr.append(
# logging.info('production_process:%s' % p.name) item.route_workcenter_id.surface_technics_id.id)
# process_parameters = production_item.product_id.model_process_parameters_ids.filtered( route_workcenter_arr.append(item.route_workcenter_id.id)
# lambda pm: pm.process_id.id == p.id) if surface_technics_arr:
# for process_parameter in process_parameters: production_process = self.env['sf.production.process'].search(
# i += 1 [('id', 'in', surface_technics_arr)],
# route_production_process = self.env[ order='sequence asc'
# 'mrp.routing.workcenter'].search( )
# [('surface_technics_id', '=', p.id), for p in production_process:
# ('id', 'in', route_workcenter_arr)]) logging.info('production_process:%s' % p.name)
# technology_design_values.append( process_parameters = production_item.product_id.model_process_parameters_ids.filtered(
# self.env['sf.technology.design'].json_technology_design_str(False, lambda pm: pm.process_id.id == p.id)
# route_production_process, for process_parameter in process_parameters:
# i, i += 1
# process_parameter)) route_production_process = self.env[
# production_item.technology_design_ids = technology_design_values 'mrp.routing.workcenter'].search(
# productions.write({'state': 'technology_to_confirmed'}) [('surface_technics_id', '=', p.id),
# return True ('id', 'in', route_workcenter_arr)])
technology_design_values.append(
self.env['sf.technology.design'].json_technology_design_str(False,
route_production_process,
i,
process_parameter))
production_item.technology_design_ids = technology_design_values
productions.write({'state': 'technology_to_confirmed'})
return True
class ProductionLot(models.Model): class ProductionLot(models.Model):
@@ -1205,20 +1216,6 @@ class ReStockMove(models.Model):
res['lot_id'] = self.subcontract_workorder_id.production_id.move_raw_ids.move_line_ids[0].lot_id.id res['lot_id'] = self.subcontract_workorder_id.production_id.move_raw_ids.move_line_ids[0].lot_id.id
return res return res
def _get_subcontract_bom(self):
self.ensure_one()
purchase_type = getattr(self.picking_id.purchase_id, 'purchase_type', False)
if purchase_type:
self = self.with_context(stock_picking=purchase_type)
bom = self.env['mrp.bom'].sudo()._bom_subcontract_find(
self.product_id,
picking_type=self.picking_type_id,
company_id=self.company_id.id,
bom_type='subcontract',
subcontractor=self.picking_id.partner_id
)
return bom
class ReStockQuant(models.Model): class ReStockQuant(models.Model):
_inherit = 'stock.quant' _inherit = 'stock.quant'

View File

@@ -2,4 +2,3 @@ from . import ftp_operate
from . import res_config_setting from . import res_config_setting
from . import sync_common from . import sync_common
from . import order_price from . import order_price
from . import mrp_production

View File

@@ -1,44 +0,0 @@
from itertools import groupby
from odoo import models, api
from odoo.tools.misc import OrderedSet
class MrpProduction(models.Model):
_inherit = 'mrp.production'
@api.model_create_multi
def create(self, vals_list):
"""
生成编程单
"""
productions = super().create(vals_list)
# 定义变量存储编程单
grouped_product_programming_no = {}
# 定义产品拼接成的制造订单名称
grouped_product_production_name = {}
# 查出所有的制造订单,为了适配通过补货生成的制造订单
all_productions = self.env['mrp.production'].search([('origin', '=', productions[0].origin)])
# 将不同产品的制造订单进行分组
grouped_product_productions = {k: list(g) for k, g in groupby(all_productions, key=lambda x: x.product_id.id)}
for product_id, grouped_productions in grouped_product_productions.items():
# 产品对应的编程单号
if product_id not in grouped_product_programming_no:
# 使用列表推导式获取非空的programming_no
programming_nos = [p.programming_no for p in grouped_productions if p.programming_no if p.programming_no is not False]
if programming_nos:
grouped_product_programming_no[product_id] = programming_nos[0]
grouped_product_production_name[product_id] = ','.join(list(map(lambda p:p.name, grouped_productions)))
# 同一个产品的制造订单只请求一次CNC编程
for production in productions:
if not production.programming_no and production.production_type in ['自动化产线加工','人工线下加工']:
if production.product_id.id not in grouped_product_programming_no:
production.fetchCNC(grouped_product_production_name[production.product_id.id])
grouped_product_programming_no[production.product_id.id] = production.programming_no
else:
production.write({
'programming_no': grouped_product_programming_no[production.product_id.id],
'programming_state': '编程中'
})
return productions

View File

@@ -3,4 +3,3 @@
from . import custom_plan from . import custom_plan
from . import change_manufactuing from . import change_manufactuing
from . import mrp_production

View File

@@ -1,41 +0,0 @@
from odoo import models, api
class MrpProduction(models.Model):
_inherit = 'mrp.production'
@api.model_create_multi
def create(self, vals_list):
"""
创建生产计划
"""
productions = super().create(vals_list)
for production in productions:
# 工单耗时
workorder_duration = 0
for workorder in production.workorder_ids:
workorder_duration += workorder.duration_expected
sale_order = self.env['sale.order'].sudo().search([('name', '=', production.origin)])
# 如果订单为空,则获取来源制造订单的销售单
if not sale_order:
mrp_production = self.env['mrp.production'].sudo().search([('name', '=', production.origin)],
limit=1)
if mrp_production:
sale_order = self.env['sale.order'].sudo().search([('name', '=', mrp_production.origin)])
else:
mrp_production = production
# if sale_order:
# sale_order.write({'schedule_status': 'to schedule'})
self.env['sf.production.plan'].sudo().with_company(production.company_id).create({
'name': production.name,
'order_deadline': sale_order.deadline_of_delivery,
'production_id': production.id,
'date_planned_start': production.date_planned_start,
'origin': mrp_production.origin,
'product_qty': production.product_qty,
'product_id': production.product_id.id,
'state': 'draft',
})
return productions

View File

@@ -200,11 +200,11 @@
<p></p> <p></p>
</div> </div>
</div> --> </div> -->
<!-- 页脚固定在底部 -->
<!-- <div style="position: absolute; bottom: 0; left: 0; right: 0;"> --> <!-- 页脚固定在底部 -->
<t t-call="sf_quality.report_quality_footer"/> <!-- <div style="position: absolute; bottom: 0; left: 0; right: 0;"> -->
<!-- </div> --> <t t-call="sf_quality.report_quality_footer"/>
<!-- </div> -->
</div> </div>
</t> </t>
</t> </t>
@@ -329,11 +329,9 @@
</div> --> </div> -->
<!-- 页脚固定在底部 --> <!-- 页脚固定在底部 -->
<!-- <t t-if="loop.index == len(docs) - 1">--> <!-- <div style="position: absolute; bottom: 0; left: 0; right: 0;"> -->
<!-- <div style="position: absolute; bottom: 0; left: 0; right: 0;"> --> <t t-call="sf_quality.html_report_quality_footer"/>
<t t-call="sf_quality.html_report_quality_footer"/> <!-- </div> -->
<!-- </div> -->
<!-- </t>-->
</div> </div>
</t> </t>
</t> </t>