@@ -165,7 +195,7 @@
+ />
diff --git a/sf_mrs_connect/models/sync_common.py b/sf_mrs_connect/models/sync_common.py
index f101530b..4926726e 100644
--- a/sf_mrs_connect/models/sync_common.py
+++ b/sf_mrs_connect/models/sync_common.py
@@ -787,6 +787,10 @@ class MachineToolType(models.Model):
result = json.loads(r['result'])
if result['status'] == 1:
for item in result['machine_tool_type_all_list']:
+ if item.get('machine_tool_picture'):
+ image = base64.b64decode(item['machine_tool_picture'])
+ else:
+ image = ''
brand = self.env['sf.machine_tool.type'].search(
[("code", '=', item['code'])])
if not brand:
@@ -811,7 +815,7 @@ class MachineToolType(models.Model):
[('code', '=', item['control_system_id'])]).id,
"active": item['active'],
'brand_id': self.env['sf.machine.brand'].search([('code', '=', item['brand_id'])]).id,
- 'machine_tool_picture': base64.b64decode(item['machine_tool_picture']),
+ 'machine_tool_picture':image,
"heightened_way": item['heightened_way'],
"workpiece_load": item['workpiece_load'],
"lead_screw": item['lead_screw'],
diff --git a/sf_plan_management/__init__.py b/sf_plan_management/__init__.py
new file mode 100644
index 00000000..9a7e03ed
--- /dev/null
+++ b/sf_plan_management/__init__.py
@@ -0,0 +1 @@
+from . import models
\ No newline at end of file
diff --git a/sf_plan_management/__manifest__.py b/sf_plan_management/__manifest__.py
new file mode 100644
index 00000000..9a8af0cd
--- /dev/null
+++ b/sf_plan_management/__manifest__.py
@@ -0,0 +1,34 @@
+# -*- coding: utf-8 -*-
+# Part of Odoo. See LICENSE file for full copyright and licensing details.
+{
+ 'name': '机企猫智能工厂 计划管理',
+ 'version': '1.0',
+ 'summary': '智能工厂计划管理',
+ 'sequence': 1,
+ 'description': """
+在本模块,定义了计划管理的清单和原型
+ """,
+ 'category': 'sf',
+ 'website': 'https://www.sf.jikimo.com',
+ 'depends': [],
+ 'data': [
+ 'security/ir.model.access.csv',
+ 'views/paln_base_view.xml',
+ 'views/menu_view.xml'
+ ],
+ 'demo': [
+ ],
+ 'assets': {
+
+ 'web.assets_qweb': [
+ ],
+ 'web.assets_backend':[
+ ]
+
+
+ },
+ 'license': 'LGPL-3',
+ 'installable': True,
+ 'application': False,
+ 'auto_install': False,
+}
diff --git a/sf_plan_management/models/__init__.py b/sf_plan_management/models/__init__.py
new file mode 100644
index 00000000..9a2c9000
--- /dev/null
+++ b/sf_plan_management/models/__init__.py
@@ -0,0 +1,2 @@
+from . import calendar_base
+from . import base
\ No newline at end of file
diff --git a/sf_plan_management/models/base.py b/sf_plan_management/models/base.py
new file mode 100644
index 00000000..76b1c0fc
--- /dev/null
+++ b/sf_plan_management/models/base.py
@@ -0,0 +1,43 @@
+from odoo import models, fields
+
+
+class ProductionLine(models.Model):
+ _name = 'sf.production.line'
+ _description = '生产线'
+
+ name = fields.Char(string='生产线名称')
+
+
+class WorkingProcedure(models.Model):
+ _name = 'sf.working.procedure'
+ _description = '工序'
+
+ name = fields.Char(string='工序名称')
+ content = fields.Char(string='主要加工内容')
+
+ WorkingProcedure_to_ProductionLine_id = fields.Many2one('sf.production.line')
+
+
+class ProcedureEquipmentResourceSetting(models.Model):
+ _name = 'sf.procedure.equipment.resource.setting'
+ _description = '工序设备资源设置'
+
+ equipment_to_working_procedure_id = fields.Many2one('sf.working.procedure', string='工序')
+ name = fields.Char(string='设备名称')
+ machine_tool_name = fields.Char(string='机台号')
+ brand = fields.Char(string='品牌')
+ model = fields.Char(string='型号')
+ production_capacity = fields.Char(string='产能')
+ working_calendar = fields.Many2one('sf.work.log.setting', string='工作日历')
+ working_shift = fields.Char(string='班次')
+ create_time = fields.Datetime(string='新增时间', default=lambda self: fields.Datetime.now())
+ stale_dated_time = fields.Datetime(string='过期时间')
+ status = fields.Selection([('0', '正常'), ('1', '故障停机'), ('2', '计划停机')], string='设备状态', default='0')
+ participate_in_scheduling = fields.Boolean(string='参与排程', default=True)
+
+
+
+
+
+
+
diff --git a/sf_plan_management/models/calendar_base.py b/sf_plan_management/models/calendar_base.py
new file mode 100644
index 00000000..dc22b92f
--- /dev/null
+++ b/sf_plan_management/models/calendar_base.py
@@ -0,0 +1,50 @@
+from odoo import models, fields, api
+import re
+
+
+class WorkLogSetting(models.Model):
+ _name = 'sf.work.log.setting'
+ _description = '工作日历设置'
+
+ name = fields.Char(string='工作日历名称')
+ # start_time = fields.Char(string='日开始时间')
+ start_time = fields.Datetime(string='日开始时间')
+ end_time = fields.Char(string='日结束时间')
+ duration = fields.Char(string='时长')
+ day_off = fields.Char(string='休息日')
+
+ user_defined_working_shift_status = fields.Boolean(string='自定义班次', default=False)
+ working_shift = fields.Char(string='班次')
+ working_shift_char = fields.Char(string='班次')
+ working_shift_select = fields.Selection([('0', '早班00:00-08:00'),
+ ('1', '白班08:00-16:00'),
+ ('2', '晚班16:00-24:00'),
+ ('3', '长白班08:00-20:00'),
+ ('4', '长晚班20:00-08:00')], string='班次')
+
+ status = fields.Boolean(string='状态', default=True)
+ update_person = fields.Char(string='更新人', default=lambda self: self.env.user.name)
+ update_time = fields.Datetime(string='更新时间', default=lambda self: fields.Datetime.now())
+
+ @api.onchange('working_shift_char', 'working_shift_select')
+ def _onchange_working_shift(self):
+ for record in self:
+ if record.working_shift_select:
+ record.working_shift = record.working_shift_select
+ else:
+ record.working_shift = record.working_shift_char
+
+ # @api.onchange('start_time')
+ # def _onchange_start_time(self):
+ # pattern = re.compile(r'^(([0-9]|1[0-9]|2[0-3]):[0-5][0-9])|24:00$')
+ # if self.start_time and not pattern.match(self.start_time):
+ # raise models.ValidationError('输入的日开始时间不正确,请重新输入!')
+
+ @api.onchange('end_time')
+ def _onchange_end_time(self):
+ pattern = re.compile(r'^(([0-9]|1[0-9]|2[0-3]):[0-5][0-9])|24:00$')
+ for record in self:
+ if record.end_time and not pattern.match(record.end_time):
+ raise models.ValidationError('输入的日结束时间不正确,请重新输入!')
+
+
diff --git a/sf_plan_management/security/ir.model.access.csv b/sf_plan_management/security/ir.model.access.csv
new file mode 100644
index 00000000..1103d658
--- /dev/null
+++ b/sf_plan_management/security/ir.model.access.csv
@@ -0,0 +1,11 @@
+id,name,model_id:id,group_id:id,perm_read,perm_write,perm_create,perm_unlink
+access_sf_work_log_setting,sf.work.log.setting,model_sf_work_log_setting,base.group_user,1,1,1,1
+access_sf_production_line,sf.production.line,model_sf_production_line,base.group_user,1,1,1,1
+access_sf_working_procedure,sf.working.procedure,model_sf_working_procedure,base.group_user,1,1,1,1
+access_sf_procedure_equipment_resource_setting,sf.procedure.equipment.resource.setting,model_sf_procedure_equipment_resource_setting,base.group_user,1,1,1,1
+
+
+
+
+
+
diff --git a/sf_plan_management/views/menu_view.xml b/sf_plan_management/views/menu_view.xml
new file mode 100644
index 00000000..1641947c
--- /dev/null
+++ b/sf_plan_management/views/menu_view.xml
@@ -0,0 +1,28 @@
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/sf_plan_management/views/paln_base_view.xml b/sf_plan_management/views/paln_base_view.xml
new file mode 100644
index 00000000..b8b63cb3
--- /dev/null
+++ b/sf_plan_management/views/paln_base_view.xml
@@ -0,0 +1,103 @@
+
+
+
+
+ 工作日历设置
+ sf.work.log.setting
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ 工作日历设置
+ sf.work.log.setting
+
+
+
+
+
+
+ 工作日历设置
+ sf.work.log.setting
+ tree,form
+
+
+
+
+
+ 工序设备资源设置
+ sf.procedure.equipment.resource.setting
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ 工序设备资源设置
+ sf.procedure.equipment.resource.setting
+ tree,form
+
+
\ No newline at end of file
diff --git a/sf_warehouse/__init__.py b/sf_warehouse/__init__.py
new file mode 100644
index 00000000..815c355c
--- /dev/null
+++ b/sf_warehouse/__init__.py
@@ -0,0 +1,3 @@
+# -*-coding:utf-8-*-
+from . import models
+
diff --git a/sf_warehouse/__manifest__.py b/sf_warehouse/__manifest__.py
new file mode 100644
index 00000000..6ad297c8
--- /dev/null
+++ b/sf_warehouse/__manifest__.py
@@ -0,0 +1,35 @@
+# -*- coding: utf-8 -*-
+# Part of Odoo. See LICENSE file for full copyright and licensing details.
+{
+ 'name': '机企猫智能工厂 库存管理',
+ 'version': '1.0',
+ 'summary': '智能工厂库存管理',
+ 'sequence': 1,
+ 'description': """
+在本模块,升级了odoo原生的库存模块
+ """,
+ 'category': 'sf',
+ 'website': 'https://www.sf.jikimo.com',
+ 'depends': ['stock', ],
+ 'data': [
+ #'security/group_security.xml',
+ # 'security/ir.model.access.csv',
+ 'views/view.xml',
+ ],
+ 'demo': [
+ ],
+ 'assets': {
+
+ 'web.assets_qweb': [
+ ],
+ 'web.assets_backend':[
+ # 'sf_tool_management/static/src/change.scss'
+ ]
+
+
+ },
+ 'license': 'LGPL-3',
+ 'installable': True,
+ 'application': False,
+ 'auto_install': False,
+}
diff --git a/sf_warehouse/models/__init__.py b/sf_warehouse/models/__init__.py
new file mode 100644
index 00000000..833a0ba1
--- /dev/null
+++ b/sf_warehouse/models/__init__.py
@@ -0,0 +1,2 @@
+from . import model
+
diff --git a/sf_warehouse/models/model.py b/sf_warehouse/models/model.py
new file mode 100644
index 00000000..cdbaded9
--- /dev/null
+++ b/sf_warehouse/models/model.py
@@ -0,0 +1,178 @@
+# -*- coding: utf-8 -*-
+from odoo import fields, models, api
+from odoo.exceptions import ValidationError, UserError
+
+
+class SfLocation(models.Model):
+ _inherit = 'stock.location'
+
+ # 重写字段定义
+ name = fields.Char('Location Name', required=True, size=20)
+ barcode = fields.Char('Barcode', copy=False, required=True, size=15)
+
+ # 仓库类别(selection:仓库、库区、库位、货位)
+ location_type = fields.Selection([
+ ('仓库', '仓库'),
+ ('库区', '库区'),
+ ('货架', '货架'),
+ ('货位', '货位')
+ ], string='仓库类别')
+ # 仓库类型(分类:成品库、坯料库、原材料库、刀具库、线边料库、线边刀库)
+ location_category = fields.Selection([
+ ('成品库', '成品库'),
+ ('坯料库', '坯料库'),
+ ('原材料库', '原材料库'),
+ ('刀具库', '刀具库'),
+ ('线边料库', '线边料库'),
+ ('线边刀库', '线边刀库')
+ ], string='仓库类型')
+ # 库区类型(selection:拣货区、存货区、收货区、退货区、次品区)
+ area_type = fields.Selection([
+ ('拣货区', '拣货区'),
+ ('存货区', '存货区'),
+ ('收货区', '收货区'),
+ ('退货区', '退货区'),
+ ('次品区', '次品区')
+ ], string='库区类型')
+ # 货架独有字段:通道、方向、货架高度(m)、货架层数、层数容量
+ channel = fields.Char(string='通道', required=True)
+ direction = fields.Selection([
+ ('R', 'R'),
+ ('L', 'L')
+ ], string='方向', required=True)
+ shelf_height = fields.Float(string='货架高度(m)')
+ shelf_layer = fields.Integer(string='货架层数', required=True)
+ layer_capacity = fields.Integer(string='层数容量', required=True)
+
+ # 货位独有字段:货位状态、产品(关联产品对象)、产品序列号(关联产品序列号对象)
+ location_status = fields.Selection([
+ ('空闲', '空闲'),
+ ('占用', '占用'),
+ ('禁用', '禁用')
+ ], string='货位状态', default='空闲')
+ # product_id = fields.Many2one('product.template', string='产品')
+ product_id = fields.Many2one('product.product', string='产品', compute='_compute_product_id', readonly=True)
+ product_sn_id = fields.Many2one('stock.lot', string='产品序列号')
+
+ # 添加SQL约束
+ _sql_constraints = [
+ ('name_uniq', 'unique(name)', '位置名称必须唯一!'),
+ ]
+
+ hide_location_type = fields.Boolean(compute='_compute_hide_what', string='隐藏仓库')
+ hide_area = fields.Boolean(compute='_compute_hide_what', string='隐藏库区')
+ hide_shelf = fields.Boolean(compute='_compute_hide_what', string='隐藏货架')
+ hide_location = fields.Boolean(compute='_compute_hide_what', string='隐藏货位')
+
+ # @api.constrains('shelf_height')
+ # def _check_shelf_height(self):
+ # for record in self:
+ # if not (0 <= record.shelf_height < 1000): # 限制字段值在0到999之间
+ # raise UserError('shelf_height的值必须在0到1000之间')
+ #
+ # @api.constrains('shelf_layer')
+ # def _check_shelf_layer(self):
+ # for record in self:
+ # if not (0 < record.shelf_layer < 1000):
+ # raise UserError('shelf_layer的值必须在0到999之间,且不能为0')
+ #
+ # @api.constrains('layer_capacity')
+ # def _check_layer_capacity(self):
+ # for record in self:
+ # if not (0 <= record.layer_capacity < 1000):
+ # raise UserError('layer_capacity的值必须在0到999之间,且不能为0')
+
+ @api.depends('product_sn_id')
+ def _compute_product_id(self):
+ for record in self:
+ if record.product_sn_id:
+ record.product_id = record.product_sn_id.product_id
+ record.location_status = '占用'
+ else:
+ record.product_id = False
+ record.location_status = '空闲'
+
+ @api.depends('location_type')
+ def _compute_hide_what(self):
+ """
+ 根据仓库类别,隐藏不需要的字段
+ :return:
+ """
+ for record in self:
+ record.hide_location_type = False
+ record.hide_area = False
+ record.hide_shelf = False
+ record.hide_location = False
+ if record.location_type and record.location_type == '仓库':
+ record.hide_location_type = True
+ elif record.location_type and record.location_type == '库区':
+ record.hide_area = True
+ elif record.location_type and record.location_type == '货架':
+ record.hide_shelf = True
+ elif record.location_type and record.location_type == '货位':
+ record.hide_location = True
+ else:
+ pass
+
+ # # 添加Python约束
+ # @api.constrains('name', 'barcode')
+ # def _check_len(self):
+ # for rec in self:
+ # if len(rec.name) > 20:
+ # raise ValidationError("Location Name length must be less equal than 20!")
+ # if len(rec.barcode) > 15:
+ # raise ValidationError("Barcode length must be less equal than 15!")
+
+ # @api.model
+ # def default_get(self, fields):
+ # print('fields:', fields)
+ # res = super(SfLocation, self).default_get(fields)
+ # print('res:', res)
+ # if 'barcode' in fields and 'barcode' not in res:
+ # # 这里是你生成barcode的代码
+ # pass
+ # # res['barcode'] = self.generate_barcode() # 假设你有一个方法generate_barcode来生成barcode
+ # return res
+ # @api.model
+ # def create(self, vals):
+ # """
+ # 重写create方法,当仓库类型为货架时,自动生成其下面的货位,数量为货架层数*层数容量
+ # """
+ # res = super(SfLocation, self).create(vals)
+ # if res.location_type == '货架':
+ # for i in range(res.shelf_layer):
+ # for j in range(res.layer_capacity):
+ # self.create({
+ # 'name': res.name + '-' + str(i+1) + '-' + str(j+1),
+ # 'location_id': res.id,
+ # 'location_type': '货位',
+ # 'barcode': self.generate_barcode(res, i, j),
+ # 'location_status': '空闲'
+ # })
+ # return res
+
+ # 生成货位
+ def create_location(self):
+ if self.location_type == '货架':
+ for i in range(self.shelf_layer):
+ for j in range(self.layer_capacity):
+ self.create({
+ 'name': self.name + '-' + str(i + 1) + '层' + '-' + str(j + 1) + '位置',
+ 'location_id': self.id,
+ 'location_type': '货位',
+ 'barcode': self.generate_barcode(i, j),
+ 'location_status': '空闲'
+ })
+
+ def generate_barcode(self, i, j):
+ # 这里是你生成barcode的代码
+ area_type_barcode = self.location_id.barcode
+ i_str = str(i + 1).zfill(3) # 确保是两位数,如果不足两位,左侧补0
+ j_str = str(j + 1).zfill(3) # 确保是两位数,如果不足两位,左侧补0
+ return area_type_barcode + self.channel + self.direction + '-' + self.barcode + '-' + i_str + '-' + j_str
+
+
+ # def generate_barcode(self, i, j):
+ # # 这里是你生成barcode的代码
+ # area_type_barcode = self.location_id.barcode
+ # return area_type_barcode + self.channel + self.direction + '-' + self.barcode + '-' + str(i + 1) + '-' + str(j + 1)
diff --git a/sf_warehouse/security/ir.model.access.csv b/sf_warehouse/security/ir.model.access.csv
new file mode 100644
index 00000000..94ce7d2b
--- /dev/null
+++ b/sf_warehouse/security/ir.model.access.csv
@@ -0,0 +1,4 @@
+id,name,model_id:id,group_id:id,perm_read,perm_write,perm_create,perm_unlink
+access_stock_location,stock.location,model_stock_location,base.group_user,1,1,1,1
+
+
diff --git a/sf_warehouse/static/src/change.scss b/sf_warehouse/static/src/change.scss
new file mode 100644
index 00000000..40fdcc8e
--- /dev/null
+++ b/sf_warehouse/static/src/change.scss
@@ -0,0 +1,11 @@
+.modal-content .o_cp_buttons {
+ display:none
+}
+
+.modal-content .o_control_panel {
+ display:none
+}
+
+.modal-content .o_list_button {
+
+}
\ No newline at end of file
diff --git a/sf_warehouse/views/view.xml b/sf_warehouse/views/view.xml
new file mode 100644
index 00000000..e2b4a3b1
--- /dev/null
+++ b/sf_warehouse/views/view.xml
@@ -0,0 +1,129 @@
+
+
+
+
+ stock.location.tree.sf.inherit
+ stock.location
+
+
+
+
+
+
+
+
+
+ stock.location.form.sf.inherit
+ stock.location
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ stock.location.search.sf.inherit
+ stock.location
+
+
+
+
+
+
+
+
+
+
+
+ example.kanban
+ stock.location
+
+
+
+
+
+
+
+
+
+
+
+
+
+ |
+
+
+
+
+
+
+
+
+
+
+ 货位状态
+ ir.actions.act_window
+ stock.location
+ kanban,form
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+