Compare commits
2 Commits
feature/69
...
feature/pl
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
c5dfb50da5 | ||
|
|
1f938292f5 |
@@ -20,7 +20,7 @@
|
|||||||
'version': '0.1',
|
'version': '0.1',
|
||||||
|
|
||||||
# any module necessary for this one to work correctly
|
# any module necessary for this one to work correctly
|
||||||
'depends': ['base', 'account', 'l10n_cn'],
|
'depends': ['base', 'account'],
|
||||||
|
|
||||||
# always loaded
|
# always loaded
|
||||||
'data': [
|
'data': [
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
from odoo import models, fields, api, _
|
from odoo import models, fields, api
|
||||||
|
|
||||||
from odoo.exceptions import ValidationError
|
from odoo.exceptions import ValidationError
|
||||||
|
|
||||||
@@ -7,14 +7,6 @@ class CustomAccountMoveLine(models.Model):
|
|||||||
_inherit = 'account.move'
|
_inherit = 'account.move'
|
||||||
_description = "account move line"
|
_description = "account move line"
|
||||||
|
|
||||||
fapiao = fields.Char(string='发票号', size=20, copy=False, tracking=True, required=True)
|
|
||||||
|
|
||||||
@api.constrains('fapiao')
|
|
||||||
def _check_fapiao(self):
|
|
||||||
for record in self:
|
|
||||||
if record.fapiao and (len(record.fapiao) != 20 or not record.fapiao.isdecimal()):
|
|
||||||
raise ValidationError(_("Fapiao number is an 20-digit number. Please enter a correct one."))
|
|
||||||
|
|
||||||
@api.model_create_multi
|
@api.model_create_multi
|
||||||
def create(self, vals):
|
def create(self, vals):
|
||||||
for val in vals:
|
for val in vals:
|
||||||
|
|||||||
@@ -21,8 +21,8 @@
|
|||||||
'web.assets_qweb': [
|
'web.assets_qweb': [
|
||||||
],
|
],
|
||||||
'web.assets_backend': [
|
'web.assets_backend': [
|
||||||
# 'jikimo_frontend/static/src/fields/custom_many2many_checkboxes/*',
|
'jikimo_frontend/static/src/fields/custom_many2many_checkboxes/*',
|
||||||
# 'jikimo_frontend/static/src/fields/Many2OneRadioField/*',
|
'jikimo_frontend/static/src/fields/Many2OneRadioField/*',
|
||||||
# 移除odoo相关标识
|
# 移除odoo相关标识
|
||||||
'jikimo_frontend/static/src/bye_odoo/*',
|
'jikimo_frontend/static/src/bye_odoo/*',
|
||||||
'jikimo_frontend/static/src/scss/custom_style.scss',
|
'jikimo_frontend/static/src/scss/custom_style.scss',
|
||||||
|
|||||||
@@ -1,8 +1,3 @@
|
|||||||
.o_list_renderer .o_list_table tbody > tr > td:not(.o_list_record_selector):not(.o_handle_cell):not(.o_list_button):not(.o_list_record_remove){
|
.o_list_renderer .o_list_table tbody > tr > td:not(.o_list_record_selector):not(.o_handle_cell):not(.o_list_button):not(.o_list_record_remove){
|
||||||
border:1px solid #dee2e6 !important;
|
border:1px solid #dee2e6 !important;
|
||||||
}
|
|
||||||
|
|
||||||
.custom_required_add::before{
|
|
||||||
content: '*';
|
|
||||||
color: red;
|
|
||||||
}
|
}
|
||||||
@@ -0,0 +1,3 @@
|
|||||||
|
.many2one_radio_field {
|
||||||
|
display: inline-block;
|
||||||
|
}
|
||||||
@@ -0,0 +1,53 @@
|
|||||||
|
/** @odoo-module **/
|
||||||
|
|
||||||
|
import { RadioField } from "@web/views/fields/radio/radio_field"; // 导入单选按钮组件
|
||||||
|
import { registry } from "@web/core/registry";
|
||||||
|
|
||||||
|
export class Many2OneRadioField extends RadioField {
|
||||||
|
// 你可以重写或者添加一些方法和属性
|
||||||
|
// 例如,你可以重写setup方法来添加一些事件监听器或者初始化一些变量
|
||||||
|
setup() {
|
||||||
|
super.setup(); // 调用父类的setup方法
|
||||||
|
// 你自己的代码
|
||||||
|
}
|
||||||
|
|
||||||
|
onImageClick(event) {
|
||||||
|
// 放大图片逻辑
|
||||||
|
// 获取图片元素
|
||||||
|
const img = event.target;
|
||||||
|
const close = img.nextSibling;
|
||||||
|
// 实现放大图片逻辑
|
||||||
|
// 比如使用 CSS 放大
|
||||||
|
img.parentElement.classList.add('zoomed');
|
||||||
|
close.classList.add('img_close');
|
||||||
|
}
|
||||||
|
|
||||||
|
onCloseClick(event) {
|
||||||
|
const close = event.target;
|
||||||
|
const img = close.previousSibling;
|
||||||
|
img.parentElement.classList.remove('zoomed');
|
||||||
|
close.classList.remove('img_close');
|
||||||
|
}
|
||||||
|
|
||||||
|
get items() {
|
||||||
|
return Many2OneRadioField.getItems(this.props.name, this.props.record);
|
||||||
|
}
|
||||||
|
|
||||||
|
static getItems(fieldName, record) {
|
||||||
|
switch (record.fields[fieldName].type) {
|
||||||
|
case "selection":
|
||||||
|
return record.fields[fieldName].selection;
|
||||||
|
case "many2one": {
|
||||||
|
const value = record.preloadedData[fieldName] || [];
|
||||||
|
return value.map((item) => [item.id, item.display_name, item.image]);
|
||||||
|
}
|
||||||
|
default:
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Many2OneRadioField.template = "jikimo_frontend.Many2OneRadioField";
|
||||||
|
// MyCustomWidget.supportedTypes = ['many2many'];
|
||||||
|
|
||||||
|
registry.category("fields").add("many2one_radio", Many2OneRadioField);
|
||||||
@@ -0,0 +1,35 @@
|
|||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<templates xml:space="preserve">
|
||||||
|
|
||||||
|
<t t-name="jikimo_frontend.Many2OneRadioField" owl="1">
|
||||||
|
<div
|
||||||
|
role="radiogroup"
|
||||||
|
t-attf-class="o_{{ props.orientation }}"
|
||||||
|
t-att-aria-label="string"
|
||||||
|
>
|
||||||
|
<t t-foreach="items" t-as="item" t-key="item[0]">
|
||||||
|
<div class="form-check o_radio_item many2one_radio_field" aria-atomic="true">
|
||||||
|
<input
|
||||||
|
type="radio"
|
||||||
|
class="form-check-input o_radio_input"
|
||||||
|
t-att-checked="item[0] === value"
|
||||||
|
t-att-disabled="props.readonly"
|
||||||
|
t-att-name="id"
|
||||||
|
t-att-data-value="item[0]"
|
||||||
|
t-att-data-index="item_index"
|
||||||
|
t-att-id="`${id}_${item[0]}`"
|
||||||
|
t-on-change="() => this.onChange(item)"
|
||||||
|
/>
|
||||||
|
<label class="form-check-label o_form_label" t-att-for="`${id}_${item[0]}`" t-esc="item[1]" />
|
||||||
|
<div t-on-dblclick="onImageClick">
|
||||||
|
<t>
|
||||||
|
<img t-att-src="item[2]" width="50" height="50"/>
|
||||||
|
<div class="close" t-on-click="onCloseClick">×</div>
|
||||||
|
</t>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</t>
|
||||||
|
</div>
|
||||||
|
</t>
|
||||||
|
|
||||||
|
</templates>
|
||||||
@@ -0,0 +1,99 @@
|
|||||||
|
|
||||||
|
.processing-capabilities-grid {
|
||||||
|
display: grid;
|
||||||
|
grid-template-columns: repeat(6, 1fr);
|
||||||
|
gap: 10px;
|
||||||
|
width: 100%;
|
||||||
|
}
|
||||||
|
|
||||||
|
.grid-item {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.item-content {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
align-items: center;
|
||||||
|
text-align: center;
|
||||||
|
}
|
||||||
|
/*控制图片大小*/
|
||||||
|
.item-icon {
|
||||||
|
width: 50px;
|
||||||
|
height: 50px;
|
||||||
|
margin-bottom: 5px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.item-label {
|
||||||
|
font-size: 12px;
|
||||||
|
word-break: break-word;
|
||||||
|
}
|
||||||
|
|
||||||
|
@media (max-width: 1200px) {
|
||||||
|
.processing-capabilities-grid {
|
||||||
|
grid-template-columns: repeat(4, 1fr);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@media (max-width: 768px) {
|
||||||
|
.processing-capabilities-grid {
|
||||||
|
grid-template-columns: repeat(3, 1fr);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@media (max-width: 480px) {
|
||||||
|
.processing-capabilities-grid {
|
||||||
|
grid-template-columns: repeat(2, 1fr);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.image-preview-container {
|
||||||
|
position: fixed;
|
||||||
|
top: 0;
|
||||||
|
left: 0;
|
||||||
|
width: 100%;
|
||||||
|
height: 100%;
|
||||||
|
background-color: rgba(0, 0, 0, 0.9);
|
||||||
|
display: flex;
|
||||||
|
justify-content: center;
|
||||||
|
align-items: center;
|
||||||
|
z-index: 1000;
|
||||||
|
opacity: 0;
|
||||||
|
transition: opacity 0.3s ease;
|
||||||
|
}
|
||||||
|
|
||||||
|
.image-preview-container.show {
|
||||||
|
opacity: 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
.image-preview {
|
||||||
|
max-width: 90%;
|
||||||
|
max-height: 90%;
|
||||||
|
object-fit: contain;
|
||||||
|
box-shadow: 0 0 20px rgba(255, 255, 255, 0.2);
|
||||||
|
border-radius: 5px;
|
||||||
|
transform: scale(0.9);
|
||||||
|
transition: transform 0.3s ease;
|
||||||
|
}
|
||||||
|
|
||||||
|
.image-preview-container.show .image-preview {
|
||||||
|
transform: scale(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
.image-preview-close {
|
||||||
|
position: absolute;
|
||||||
|
top: 20px;
|
||||||
|
right: 30px;
|
||||||
|
color: #fff;
|
||||||
|
font-size: 40px;
|
||||||
|
font-weight: bold;
|
||||||
|
transition: 0.3s;
|
||||||
|
cursor: pointer;
|
||||||
|
opacity: 0.7;
|
||||||
|
}
|
||||||
|
|
||||||
|
.image-preview-close:hover,
|
||||||
|
.image-preview-close:focus {
|
||||||
|
opacity: 1;
|
||||||
|
text-decoration: none;
|
||||||
|
cursor: pointer;
|
||||||
|
}
|
||||||
@@ -0,0 +1,60 @@
|
|||||||
|
/** @odoo-module **/
|
||||||
|
|
||||||
|
import {Many2ManyCheckboxesField} from "@web/views/fields/many2many_checkboxes/many2many_checkboxes_field";
|
||||||
|
import {registry} from "@web/core/registry";
|
||||||
|
|
||||||
|
export class MyCustomWidget extends Many2ManyCheckboxesField {
|
||||||
|
setup() {
|
||||||
|
super.setup();
|
||||||
|
}
|
||||||
|
|
||||||
|
onImageClick(event, src) {
|
||||||
|
event.preventDefault();
|
||||||
|
event.stopPropagation();
|
||||||
|
|
||||||
|
// 创建预览框
|
||||||
|
const previewContainer = document.createElement('div');
|
||||||
|
previewContainer.className = 'image-preview-container';
|
||||||
|
|
||||||
|
const previewImg = document.createElement('img');
|
||||||
|
previewImg.src = src;
|
||||||
|
previewImg.className = 'image-preview';
|
||||||
|
// 设置放大的预览图片大小
|
||||||
|
previewImg.style.width = '600px';
|
||||||
|
previewImg.style.height = 'auto'; // 保持宽高比
|
||||||
|
|
||||||
|
const closeButton = document.createElement('span');
|
||||||
|
closeButton.innerHTML = '×';
|
||||||
|
closeButton.className = 'image-preview-close';
|
||||||
|
|
||||||
|
previewContainer.appendChild(previewImg);
|
||||||
|
previewContainer.appendChild(closeButton);
|
||||||
|
document.body.appendChild(previewContainer);
|
||||||
|
|
||||||
|
// 添加关闭预览的事件监听器
|
||||||
|
const closePreview = () => {
|
||||||
|
previewContainer.classList.remove('show');
|
||||||
|
setTimeout(() => {
|
||||||
|
document.body.removeChild(previewContainer);
|
||||||
|
}, 300);
|
||||||
|
};
|
||||||
|
|
||||||
|
closeButton.addEventListener('click', closePreview);
|
||||||
|
|
||||||
|
// 点击预览框外部也可以关闭
|
||||||
|
previewContainer.addEventListener('click', (e) => {
|
||||||
|
if (e.target === previewContainer) {
|
||||||
|
closePreview();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// 使用 setTimeout 来触发过渡效果
|
||||||
|
setTimeout(() => {
|
||||||
|
previewContainer.classList.add('show');
|
||||||
|
}, 10);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
MyCustomWidget.template = "jikimo_frontend.MyCustomWidget";
|
||||||
|
|
||||||
|
registry.category("fields").add("custom_many2many_checkboxes", MyCustomWidget);
|
||||||
@@ -0,0 +1,23 @@
|
|||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<templates xml:space="preserve">
|
||||||
|
|
||||||
|
<t t-name="jikimo_frontend.MyCustomWidget" owl="1">
|
||||||
|
<div aria-atomic="true" class="many2many_flex processing-capabilities-grid">
|
||||||
|
<t t-foreach="items" t-as="item" t-key="item[0]">
|
||||||
|
<div class="grid-item">
|
||||||
|
<CheckBox
|
||||||
|
value="isSelected(item)"
|
||||||
|
disabled="props.readonly"
|
||||||
|
onChange="(ev) => this.onChange(item[0], ev)"
|
||||||
|
>
|
||||||
|
<div class="item-content">
|
||||||
|
<img t-att-src="item[2]" class="item-icon" t-on-click="(ev) => this.onImageClick(ev, item[2])"/>
|
||||||
|
<span class="item-label"><t t-esc="item[1]"/></span>
|
||||||
|
</div>
|
||||||
|
</CheckBox>
|
||||||
|
</div>
|
||||||
|
</t>
|
||||||
|
</div>
|
||||||
|
</t>
|
||||||
|
|
||||||
|
</templates>
|
||||||
@@ -6,9 +6,8 @@ import {_t} from "@web/core/l10n/translation";
|
|||||||
import {FormStatusIndicator} from "@web/views/form/form_status_indicator/form_status_indicator";
|
import {FormStatusIndicator} from "@web/views/form/form_status_indicator/form_status_indicator";
|
||||||
import {ListRenderer} from "@web/views/list/list_renderer";
|
import {ListRenderer} from "@web/views/list/list_renderer";
|
||||||
// import {StatusBarField} from "@web/views/fields/statusbar/statusbar_field";
|
// import {StatusBarField} from "@web/views/fields/statusbar/statusbar_field";
|
||||||
import {FormLabel} from "@web/views/form/form_label";
|
|
||||||
import { fieldVisualFeedback } from "@web/views/fields/field";
|
|
||||||
|
|
||||||
|
import {Field} from "@web/views/fields/field";
|
||||||
|
|
||||||
var Dialog = require('web.Dialog');
|
var Dialog = require('web.Dialog');
|
||||||
// var {patch} = require("web.utils") 这句话也行
|
// var {patch} = require("web.utils") 这句话也行
|
||||||
@@ -52,6 +51,7 @@ const tableRequiredList = [
|
|||||||
'product_template_id', 'product_uom_qty', 'price_unit','product_id','product_qty',
|
'product_template_id', 'product_uom_qty', 'price_unit','product_id','product_qty',
|
||||||
'name', 'fault_type', 'maintenance_standards', 'Period'
|
'name', 'fault_type', 'maintenance_standards', 'Period'
|
||||||
]
|
]
|
||||||
|
|
||||||
patch(FormStatusIndicator.prototype, 'jikimo_frontend.FormStatusIndicator', {
|
patch(FormStatusIndicator.prototype, 'jikimo_frontend.FormStatusIndicator', {
|
||||||
setup() {
|
setup() {
|
||||||
owl.onMounted(() => {
|
owl.onMounted(() => {
|
||||||
@@ -62,7 +62,7 @@ patch(FormStatusIndicator.prototype, 'jikimo_frontend.FormStatusIndicator', {
|
|||||||
const dom1 = buttonsDom.children('.o_form_button_save')
|
const dom1 = buttonsDom.children('.o_form_button_save')
|
||||||
const dom2 = buttonsDom.children('.o_form_button_cancel')
|
const dom2 = buttonsDom.children('.o_form_button_cancel')
|
||||||
dom1.append('保存')
|
dom1.append('保存')
|
||||||
dom2.append('不保存')
|
dom2.append('取消')
|
||||||
}
|
}
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
console.log(e)
|
console.log(e)
|
||||||
@@ -107,37 +107,39 @@ patch(FormStatusIndicator.prototype, 'jikimo_frontend.FormStatusIndicator', {
|
|||||||
|
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
|
patch(Field.prototype, 'jikimo_frontend.Field', {
|
||||||
|
setup() {
|
||||||
|
owl.onMounted(this.setRequired);
|
||||||
|
return this._super(...arguments);
|
||||||
|
},
|
||||||
|
setRequired() {
|
||||||
|
const id = this.props.id
|
||||||
|
const isRequired = filedRequiredList[id]
|
||||||
|
if(id == 'number_of_axles') {
|
||||||
|
console.log(isRequired)
|
||||||
|
}
|
||||||
|
if(isRequired) {
|
||||||
|
let dom;
|
||||||
|
dom = $(`label[for=${id}]`)
|
||||||
|
if(isRequired.multiple && dom.length > 1) {
|
||||||
|
dom = dom.eq(-1)
|
||||||
|
dom = dom.parent().parent().next().find('label')
|
||||||
|
}
|
||||||
|
if(isRequired.noLabel) {
|
||||||
|
dom = dom.parent().parent()
|
||||||
|
}
|
||||||
|
let t = dom.html()
|
||||||
|
t = '<i class="c* r" style="color: red;margin-left: -4px">*</i>' + t
|
||||||
|
dom.html(t)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
})
|
||||||
patch(ListRenderer.prototype, 'jikimo_frontend.ListRenderer', {
|
patch(ListRenderer.prototype, 'jikimo_frontend.ListRenderer', {
|
||||||
setup(){
|
setup(){
|
||||||
owl.onMounted(() => {
|
owl.onMounted(() => {
|
||||||
this.activeElement = this.uiService.activeElement;
|
this.activeElement = this.uiService.activeElement;
|
||||||
this.setRequired()
|
this.setRequired()
|
||||||
this.listherHeaderBodyNum()
|
|
||||||
})
|
})
|
||||||
owl.onPatched(() => {
|
|
||||||
this.listherHeaderBodyNum()
|
|
||||||
})
|
|
||||||
const treeModifiers = this.getFieldModifiers(this.props.archInfo.__rawArch);
|
|
||||||
|
|
||||||
if(treeModifiers) {
|
|
||||||
if(treeModifiers.merge_fields) {
|
|
||||||
this.props.merge_key = treeModifiers.merge_key;
|
|
||||||
this.props.merge_fields = treeModifiers.merge_fields.split(',');
|
|
||||||
const data = this.setColumns(this.props.merge_key);
|
|
||||||
owl.onMounted(() => {
|
|
||||||
this.mergeColumns(this.props.merge_fields, data)
|
|
||||||
})
|
|
||||||
}
|
|
||||||
if(treeModifiers.pacthResize) {
|
|
||||||
|
|
||||||
owl.onPatched(() => {
|
|
||||||
this.columnWidths = null;
|
|
||||||
this.freezeColumnWidths();
|
|
||||||
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return this._super(...arguments);
|
return this._super(...arguments);
|
||||||
},
|
},
|
||||||
setRequired() {
|
setRequired() {
|
||||||
@@ -163,127 +165,40 @@ patch(ListRenderer.prototype, 'jikimo_frontend.ListRenderer', {
|
|||||||
} catch (e) {
|
} catch (e) {
|
||||||
console.log(e)
|
console.log(e)
|
||||||
}
|
}
|
||||||
},
|
|
||||||
listherHeaderBodyNum() {
|
|
||||||
const dom = this.tableRef.el
|
|
||||||
try {
|
|
||||||
const thead = $(dom).children('thead')
|
|
||||||
const tbody = $(dom).children('tbody')
|
|
||||||
const thead_tr = thead.children().eq(0)
|
|
||||||
const tbody_tr = tbody.children().eq(0)
|
|
||||||
const thead_th_num = thead_tr.children().length
|
|
||||||
const tbody_tr_num = tbody_tr.children().length
|
|
||||||
const num = thead_th_num - tbody_tr_num
|
|
||||||
if(num == -1) {
|
|
||||||
tbody.children('tr').each(function () {
|
|
||||||
$(this).children('td').eq(0).remove()
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
} catch (e) {
|
|
||||||
console.log(e)
|
|
||||||
}
|
|
||||||
},
|
|
||||||
setColumns( merge_key) {
|
|
||||||
merge_key = merge_key.split(',')
|
|
||||||
const data = this.props.list.records
|
|
||||||
let sourceIndex = 0;
|
|
||||||
let sourceValue = merge_key.reduce((acc, key) => {
|
|
||||||
acc[key] = '';
|
|
||||||
return acc;
|
|
||||||
}, {});
|
|
||||||
|
|
||||||
data.forEach((item, index) => {
|
|
||||||
if(!item.colspan) {
|
|
||||||
item.colspan = 1;
|
|
||||||
}
|
|
||||||
const itemValue = merge_key.reduce((acc, key) => {
|
|
||||||
acc[key] = item.data[key];
|
|
||||||
return acc;
|
|
||||||
}, {});
|
|
||||||
if(JSON.stringify(itemValue) == JSON.stringify(sourceValue)) {
|
|
||||||
data[sourceIndex].colspan ++ ;
|
|
||||||
item.colspan = 0;
|
|
||||||
} else {
|
|
||||||
sourceIndex = index;
|
|
||||||
sourceValue = itemValue;
|
|
||||||
}
|
|
||||||
})
|
|
||||||
return data
|
|
||||||
},
|
|
||||||
getFieldModifiers(xmlString) {
|
|
||||||
const parser = new DOMParser();
|
|
||||||
const xmlDoc = parser.parseFromString(xmlString, "text/xml");
|
|
||||||
|
|
||||||
// 提取 <tree> 的 modifiers
|
|
||||||
const treeElement = xmlDoc.querySelector("tree");
|
|
||||||
const treeModifiers = treeElement.getAttribute("modifiers");
|
|
||||||
if (treeModifiers) {
|
|
||||||
const parsedTreeModifiers = JSON.parse(treeModifiers);
|
|
||||||
console.log("Tree Modifiers:", parsedTreeModifiers);
|
|
||||||
return parsedTreeModifiers;
|
|
||||||
}
|
|
||||||
return null;
|
|
||||||
},
|
|
||||||
mergeColumns(merge_fields, data) {
|
|
||||||
const dom = this.tableRef.el
|
|
||||||
const thead = $(dom).children('thead')
|
|
||||||
const tbody = $(dom).children('tbody')
|
|
||||||
let row_no = 0
|
|
||||||
tbody.children('tr.o_data_row').each(function () {
|
|
||||||
const tr = $(this)
|
|
||||||
const td = tr.children('td')
|
|
||||||
const index = $(this).index()
|
|
||||||
const col = data[index].colspan
|
|
||||||
row_no ++
|
|
||||||
if(col == 0) {
|
|
||||||
row_no --
|
|
||||||
}
|
|
||||||
td.eq(0).text(row_no).attr('rowspan', col)
|
|
||||||
|
|
||||||
if(col == 0) {
|
|
||||||
td.eq(0).remove()
|
|
||||||
}
|
|
||||||
td.each(function () {
|
|
||||||
if(merge_fields.indexOf($(this).attr('name')) >= 0) {
|
|
||||||
$(this).attr('rowspan', col)
|
|
||||||
if(col == 0) {
|
|
||||||
$(this).remove()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
})
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
})
|
|
||||||
patch(FormLabel.prototype, 'jikimo_frontend.FormLabel', {
|
|
||||||
get className() {
|
|
||||||
|
|
||||||
const { invalid, empty, readonly } = fieldVisualFeedback(
|
|
||||||
this.props.fieldInfo.FieldComponent,
|
|
||||||
this.props.record,
|
|
||||||
this.props.fieldName,
|
|
||||||
this.props.fieldInfo
|
|
||||||
);
|
|
||||||
const classes = this.props.className ? [this.props.className] : [];
|
|
||||||
const otherRequired = filedRequiredList[this.props.fieldName]
|
|
||||||
if(this.props.fieldInfo?.rawAttrs?.class?.indexOf('custom_required') >= 0 || otherRequired) {
|
|
||||||
classes.push('custom_required_add')
|
|
||||||
}
|
|
||||||
if (invalid) {
|
|
||||||
classes.push("o_field_invalid");
|
|
||||||
}
|
|
||||||
if (empty) {
|
|
||||||
classes.push("o_form_label_empty");
|
|
||||||
}
|
|
||||||
if (readonly) {
|
|
||||||
classes.push("o_form_label_readonly");
|
|
||||||
}
|
|
||||||
return classes.join(" ");
|
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
|
|
||||||
|
// 根据进度条设置水印
|
||||||
|
// const statusbar_params = {
|
||||||
|
// '已完工': 'bg-primary',
|
||||||
|
// '完成': 'bg-primary',
|
||||||
|
// '采购订单': 'bg-primary',
|
||||||
|
// '作废': 'bg-danger',
|
||||||
|
// '封存(报废)': 'bg-danger',
|
||||||
|
// }
|
||||||
|
// patch(StatusBarField.prototype, 'jikimo_frontend.StatusBarField', {
|
||||||
|
// setup() {
|
||||||
|
// owl.onMounted(this.ribbons);
|
||||||
|
// return this._super(...arguments);
|
||||||
|
// },
|
||||||
|
// ribbons() {
|
||||||
|
// try {
|
||||||
|
// const dom = $('.o_form_sheet.position-relative')
|
||||||
|
// const status = statusbar_params[this.currentName]
|
||||||
|
// if(status && dom.length) {
|
||||||
|
// dom.prepend(`<div class="o_widget o_widget_web_ribbon">
|
||||||
|
// <div class="ribbon ribbon-top-right">
|
||||||
|
// <span class="bg-opacity-75 ${status}" title="">${this.currentName}</span>
|
||||||
|
// </div>
|
||||||
|
// </div>`)
|
||||||
|
// }
|
||||||
|
// } catch (e) {
|
||||||
|
// console.log(e)
|
||||||
|
// }
|
||||||
|
// }
|
||||||
|
// })
|
||||||
|
|
||||||
$(function () {
|
$(function () {
|
||||||
|
|
||||||
function customRequired() {
|
function customRequired() {
|
||||||
@@ -292,6 +207,7 @@ $(function () {
|
|||||||
clearInterval(timer)
|
clearInterval(timer)
|
||||||
timer = setInterval(() => {
|
timer = setInterval(() => {
|
||||||
timer_count++
|
timer_count++
|
||||||
|
const dom = $('.custom_required')
|
||||||
let tableDom = $('.table_custom_required')
|
let tableDom = $('.table_custom_required')
|
||||||
if (tableDom.length) {
|
if (tableDom.length) {
|
||||||
tableDom = tableDom.eq(0).parents('tr').children('.table_custom_required')
|
tableDom = tableDom.eq(0).parents('tr').children('.table_custom_required')
|
||||||
@@ -303,6 +219,17 @@ $(function () {
|
|||||||
})
|
})
|
||||||
clearInterval(timer)
|
clearInterval(timer)
|
||||||
}
|
}
|
||||||
|
if (dom.length) {
|
||||||
|
dom.each(function () {
|
||||||
|
const requiredDom = $(this).parent().prev().find('label')
|
||||||
|
let t = requiredDom.html()
|
||||||
|
if (t && t.indexOf('c*') < 0) {
|
||||||
|
t = '<i class="c*" style="color: red;margin-left: -4px">*</i>' + t
|
||||||
|
}
|
||||||
|
requiredDom.html(t)
|
||||||
|
})
|
||||||
|
clearInterval(timer)
|
||||||
|
}
|
||||||
if (timer_count == 20) {
|
if (timer_count == 20) {
|
||||||
clearInterval(timer)
|
clearInterval(timer)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -157,11 +157,11 @@ td.o_required_modifier {
|
|||||||
color: #aaa;
|
color: #aaa;
|
||||||
}
|
}
|
||||||
|
|
||||||
// .o_kanban_primary_left {
|
.o_kanban_primary_left {
|
||||||
// display: flex;
|
display: flex;
|
||||||
// flex-direction: row-reverse;
|
flex-direction: row-reverse;
|
||||||
// justify-content: flex-start;
|
justify-content: flex-start;
|
||||||
// }
|
}
|
||||||
|
|
||||||
.o_list_view .o_list_table thead {
|
.o_list_view .o_list_table thead {
|
||||||
position: sticky;
|
position: sticky;
|
||||||
@@ -536,7 +536,3 @@ div:has(.o_required_modifier) > label::before {
|
|||||||
position: unset;
|
position: unset;
|
||||||
}
|
}
|
||||||
|
|
||||||
// 修复搜索面板checkbox样式
|
|
||||||
.o_search_panel .form-check .form-check-label span {
|
|
||||||
position: relative;
|
|
||||||
}
|
|
||||||
@@ -10,6 +10,7 @@
|
|||||||
</t>
|
</t>
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
<!-- 暂存,同一份文件中有问题,拆分后正常工作 -->
|
<!-- 暂存,同一份文件中有问题,拆分后正常工作 -->
|
||||||
|
|
||||||
<!-- <t t-name="og.web.ListRenderer.Rows" t-inherit="web.ListRenderer.Rows" t-inherit-mode="extension"> -->
|
<!-- <t t-name="og.web.ListRenderer.Rows" t-inherit="web.ListRenderer.Rows" t-inherit-mode="extension"> -->
|
||||||
|
|||||||
@@ -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(
|
||||||
_("您没有执行此操作的权限。请联系管理员"))
|
_("You are restricted from performing this operation. Please contact the administrator."))
|
||||||
else:
|
else:
|
||||||
# 如果 'access.right' 模型不存在,可以在这里定义备选逻辑
|
# 如果 'access.right' 模型不存在,可以在这里定义备选逻辑
|
||||||
# 例如,记录日志、发送通知或者简单地跳过这部分逻辑
|
# 例如,记录日志、发送通知或者简单地跳过这部分逻辑
|
||||||
|
|||||||
@@ -1,2 +0,0 @@
|
|||||||
# -*- coding: utf-8 -*-
|
|
||||||
from . import models
|
|
||||||
@@ -1,18 +0,0 @@
|
|||||||
# -*- coding: utf-8 -*-
|
|
||||||
{
|
|
||||||
'name': '机企猫 打印模块',
|
|
||||||
'version': '1.0',
|
|
||||||
'summary': """ 包含机台二维码,程序单打印等 """,
|
|
||||||
'author': '机企猫',
|
|
||||||
'website': 'https://www.jikimo.com',
|
|
||||||
'category': '机企猫',
|
|
||||||
'depends': ['sf_manufacturing', 'sf_maintenance', 'base_report_to_printer'],
|
|
||||||
'data': [
|
|
||||||
'views/maintenance_views.xml',
|
|
||||||
],
|
|
||||||
|
|
||||||
'application': True,
|
|
||||||
'installable': True,
|
|
||||||
'auto_install': False,
|
|
||||||
'license': 'LGPL-3',
|
|
||||||
}
|
|
||||||
@@ -1,5 +0,0 @@
|
|||||||
# -*- coding: utf-8 -*-
|
|
||||||
from . import jikimo_printing
|
|
||||||
from . import maintenance_printing
|
|
||||||
from . import workorder_printing
|
|
||||||
|
|
||||||
@@ -1,87 +0,0 @@
|
|||||||
from io import BytesIO
|
|
||||||
import qrcode
|
|
||||||
from reportlab.pdfgen import canvas
|
|
||||||
from reportlab.lib.pagesizes import A4
|
|
||||||
from PIL import Image
|
|
||||||
import logging
|
|
||||||
from reportlab.lib.utils import ImageReader
|
|
||||||
from odoo import models, fields, api
|
|
||||||
import base64
|
|
||||||
|
|
||||||
_logger = logging.getLogger(__name__)
|
|
||||||
|
|
||||||
class JikimoPrinting(models.AbstractModel):
|
|
||||||
_name = 'jikimo.printing'
|
|
||||||
|
|
||||||
def print_qr_code(self, data):
|
|
||||||
"""
|
|
||||||
打印二维码
|
|
||||||
"""
|
|
||||||
printer = self.env['printing.printer'].get_default()
|
|
||||||
if not printer:
|
|
||||||
_logger.error("未找到默认打印机")
|
|
||||||
return False
|
|
||||||
|
|
||||||
# 生成二维码
|
|
||||||
qr = qrcode.QRCode(version=1, box_size=10, border=5)
|
|
||||||
qr.add_data(data)
|
|
||||||
qr.make(fit=True)
|
|
||||||
qr_image = qr.make_image(fill_color="black", back_color="white")
|
|
||||||
|
|
||||||
# 将PIL Image转换为reportlab可用的格式
|
|
||||||
temp_image = BytesIO()
|
|
||||||
qr_image.save(temp_image, format="PNG")
|
|
||||||
temp_image.seek(0)
|
|
||||||
|
|
||||||
# 创建PDF
|
|
||||||
pdf_buffer = BytesIO()
|
|
||||||
c = canvas.Canvas(pdf_buffer, pagesize=A4)
|
|
||||||
|
|
||||||
# 计算位置
|
|
||||||
a4_width, a4_height = A4
|
|
||||||
qr_width = 200
|
|
||||||
qr_height = 200
|
|
||||||
x = (a4_width - qr_width) / 2
|
|
||||||
y = (a4_height - qr_height) / 2
|
|
||||||
|
|
||||||
# 直接从BytesIO绘制图片
|
|
||||||
c.drawImage(ImageReader(Image.open(temp_image)), x, y, width=qr_width, height=qr_height)
|
|
||||||
c.save()
|
|
||||||
|
|
||||||
# 获取PDF内容并打印
|
|
||||||
pdf_content = pdf_buffer.getvalue()
|
|
||||||
# _logger.info(f"打印内容: {pdf_content}")
|
|
||||||
printer.print_document(report=None, content=pdf_content, doc_format='pdf')
|
|
||||||
|
|
||||||
# 清理资源
|
|
||||||
pdf_buffer.close()
|
|
||||||
temp_image.close()
|
|
||||||
|
|
||||||
return True
|
|
||||||
|
|
||||||
def print_pdf(self, pdf_data):
|
|
||||||
"""
|
|
||||||
打印PDF
|
|
||||||
"""
|
|
||||||
printer = self.env['printing.printer'].get_default()
|
|
||||||
if not printer:
|
|
||||||
_logger.error("未找到默认打印机")
|
|
||||||
return False
|
|
||||||
|
|
||||||
pdf_data_str = pdf_data.decode('ascii', errors='ignore')
|
|
||||||
decoded_data = base64.b64decode(pdf_data_str)
|
|
||||||
|
|
||||||
# 处理二进制数据
|
|
||||||
pdf_buffer = BytesIO()
|
|
||||||
pdf_buffer.write(decoded_data)
|
|
||||||
pdf_buffer.seek(0)
|
|
||||||
|
|
||||||
# 获取PDF内容
|
|
||||||
pdf_content = pdf_buffer.getvalue()
|
|
||||||
|
|
||||||
printer.print_document(report=None, content=pdf_content, doc_format='pdf')
|
|
||||||
# 清理资源
|
|
||||||
pdf_buffer.close()
|
|
||||||
|
|
||||||
_logger.info("成功打印PDF")
|
|
||||||
return True
|
|
||||||
@@ -1,69 +0,0 @@
|
|||||||
from odoo import models, fields, api
|
|
||||||
|
|
||||||
class MaintenancePrinting(models.Model):
|
|
||||||
_inherit = 'maintenance.equipment'
|
|
||||||
|
|
||||||
def print_single_method(self):
|
|
||||||
|
|
||||||
print('self.name========== %s' % self.name)
|
|
||||||
self.ensure_one()
|
|
||||||
# maintenance_equipment_id = self.id
|
|
||||||
# # host = "192.168.50.110" # 可以根据实际情况修改
|
|
||||||
# # port = 9100 # 可以根据实际情况修改
|
|
||||||
|
|
||||||
# # 获取默认打印机配置
|
|
||||||
# printer_config = self.env['printer.configuration'].sudo().search([('model', '=', self._name)], limit=1)
|
|
||||||
# if not printer_config:
|
|
||||||
# raise UserError('请先配置打印机')
|
|
||||||
# host = printer_config.printer_id.ip_address
|
|
||||||
# port = printer_config.printer_id.port
|
|
||||||
# self.print_qr_code(maintenance_equipment_id, host, port)
|
|
||||||
|
|
||||||
# 切换成A4打印机
|
|
||||||
|
|
||||||
try:
|
|
||||||
self.env['jikimo.printing'].print_qr_code(self.MTcode)
|
|
||||||
except Exception as e:
|
|
||||||
raise UserError(f"打印失败: {str(e)}")
|
|
||||||
|
|
||||||
|
|
||||||
# def generate_zpl_code(self, code):
|
|
||||||
# """生成ZPL代码用于打印二维码标签
|
|
||||||
# Args:
|
|
||||||
# code: 需要编码的内容
|
|
||||||
# Returns:
|
|
||||||
# str: ZPL指令字符串
|
|
||||||
# """
|
|
||||||
# zpl_code = "^XA\n" # 开始ZPL格式
|
|
||||||
|
|
||||||
# # 设置打印参数
|
|
||||||
# zpl_code += "^LH0,0\n" # 设置标签起始位置
|
|
||||||
# zpl_code += "^CI28\n" # 设置中文编码
|
|
||||||
# zpl_code += "^PW400\n" # 设置打印宽度为400点
|
|
||||||
# zpl_code += "^LL300\n" # 设置标签长度为300点
|
|
||||||
|
|
||||||
# # 打印标题
|
|
||||||
# zpl_code += "^FO10,20\n" # 设置标题位置
|
|
||||||
# zpl_code += "^A0N,30,30\n" # 设置字体大小
|
|
||||||
# zpl_code += "^FD机床二维码^FS\n" # 打印标题文本
|
|
||||||
|
|
||||||
# # 打印二维码
|
|
||||||
# zpl_code += "^FO50,60\n" # 设置二维码位置
|
|
||||||
# zpl_code += f"^BQN,2,8\n" # 设置二维码参数:模式2,放大倍数8
|
|
||||||
# zpl_code += f"^FDLA,{code}^FS\n" # 二维码内容
|
|
||||||
|
|
||||||
# # 打印编码文本
|
|
||||||
# zpl_code += "^FO50,220\n" # 设置编码文本位置
|
|
||||||
# zpl_code += "^A0N,25,25\n" # 设置字体大小
|
|
||||||
# zpl_code += f"^FD编码: {code}^FS\n" # 打印编码文本
|
|
||||||
|
|
||||||
# # 打印日期
|
|
||||||
# zpl_code += "^FO50,260\n"
|
|
||||||
# zpl_code += "^A0N,20,20\n"
|
|
||||||
# zpl_code += f"^FD打印日期: {fields.Date.today()}^FS\n"
|
|
||||||
|
|
||||||
# zpl_code += "^PQ1\n" # 打印1份
|
|
||||||
# zpl_code += "^XZ\n" # 结束ZPL格式
|
|
||||||
|
|
||||||
# return zpl_code
|
|
||||||
|
|
||||||
@@ -1,31 +0,0 @@
|
|||||||
|
|
||||||
import logging
|
|
||||||
from odoo import models, fields, api
|
|
||||||
|
|
||||||
_logger = logging.getLogger(__name__)
|
|
||||||
|
|
||||||
class MrpWorkorder(models.Model):
|
|
||||||
_name = 'mrp.workorder'
|
|
||||||
_inherit = ['mrp.workorder']
|
|
||||||
|
|
||||||
def _compute_state(self):
|
|
||||||
super(MrpWorkorder, self)._compute_state()
|
|
||||||
for workorder in self:
|
|
||||||
work_ids = workorder.production_id.workorder_ids.filtered(lambda w: w.routing_type == '装夹预调' or w.routing_type == '人工线下加工')
|
|
||||||
for wo in work_ids:
|
|
||||||
if wo.state == 'ready' and not wo.production_id.product_id.is_print_program:
|
|
||||||
# 触发打印程序
|
|
||||||
pdf_data = workorder.processing_drawing
|
|
||||||
if pdf_data:
|
|
||||||
try:
|
|
||||||
# 执行打印
|
|
||||||
self.env['jikimo.printing'].print_pdf(pdf_data)
|
|
||||||
wo.production_id.product_id.is_print_program = True
|
|
||||||
except Exception as e:
|
|
||||||
_logger.error(f"工单 {wo.name} 的PDF打印失败: {str(e)}")
|
|
||||||
|
|
||||||
class ProductTemplate(models.Model):
|
|
||||||
_inherit = 'product.template'
|
|
||||||
|
|
||||||
is_print_program = fields.Boolean(string='是否打印程序', default=False)
|
|
||||||
|
|
||||||
@@ -1,19 +0,0 @@
|
|||||||
<?xml version="1.0"?>
|
|
||||||
<odoo>
|
|
||||||
<record id="sf_maintenance_equipment_view_form_qrcode_print" model="ir.ui.view">
|
|
||||||
<field name="name">sf_equipment.form</field>
|
|
||||||
<field name="model">maintenance.equipment</field>
|
|
||||||
<field name="inherit_id" ref="sf_maintenance.sf_hr_equipment_view_form"/>
|
|
||||||
<field name="arch" type="xml">
|
|
||||||
<xpath expr="//field[@name='qr_code_image']" position="after">
|
|
||||||
<label for="print_single_method"/>
|
|
||||||
<div class="col-12 col-lg-6 o_setting_box" style="white-space: nowrap">
|
|
||||||
<button type="object" class="oe_highlight" name='print_single_method' string="打印机床二维码"
|
|
||||||
attrs="{'invisible': [('equipment_type', '!=', '机床')]}"/>
|
|
||||||
</div>
|
|
||||||
</xpath>
|
|
||||||
</field>
|
|
||||||
</record>
|
|
||||||
</odoo>
|
|
||||||
|
|
||||||
|
|
||||||
@@ -1,3 +0,0 @@
|
|||||||
# -*- coding: utf-8 -*-
|
|
||||||
from . import models
|
|
||||||
from . import wizard
|
|
||||||
@@ -1,30 +0,0 @@
|
|||||||
# -*- coding: utf-8 -*-
|
|
||||||
{
|
|
||||||
'name': '机企猫 采购申请',
|
|
||||||
'version': '16.0.1.0.0',
|
|
||||||
'summary': """ 机企猫 采购申请 """,
|
|
||||||
'author': '机企猫',
|
|
||||||
'website': 'https://bfw.jikimo.com',
|
|
||||||
'category': 'purchase',
|
|
||||||
'depends': ['sf_manufacturing', 'purchase_request'],
|
|
||||||
'data': [
|
|
||||||
'security/ir.model.access.csv',
|
|
||||||
'views/sale_order_view.xml',
|
|
||||||
'views/mrp_production.xml',
|
|
||||||
'views/purchase_request_view.xml',
|
|
||||||
'wizard/purchase_request_line_make_purchase_order_view.xml',
|
|
||||||
'views/purchase_request_line_view.xml',
|
|
||||||
'views/stock_picking_views.xml',
|
|
||||||
'wizard/purchase_request_wizard_views.xml',
|
|
||||||
'views/purchase_request_menu_views.xml',
|
|
||||||
],
|
|
||||||
'assets': {
|
|
||||||
'web.assets_backend': [
|
|
||||||
'jikimo_purchase_request/static/src/**/*'
|
|
||||||
],
|
|
||||||
},
|
|
||||||
'application': True,
|
|
||||||
'installable': True,
|
|
||||||
'auto_install': False,
|
|
||||||
'license': 'LGPL-3',
|
|
||||||
}
|
|
||||||
File diff suppressed because it is too large
Load Diff
@@ -1,8 +0,0 @@
|
|||||||
# -*- coding: utf-8 -*-
|
|
||||||
from . import product_template
|
|
||||||
from . import purchase_request
|
|
||||||
from . import sale_order
|
|
||||||
from . import mrp_production
|
|
||||||
from . import purchase_order
|
|
||||||
from . import stock_rule
|
|
||||||
from . import stock_picking
|
|
||||||
@@ -1,58 +0,0 @@
|
|||||||
from odoo import fields, models, api, _
|
|
||||||
|
|
||||||
|
|
||||||
class MrpProduction(models.Model):
|
|
||||||
_inherit = 'mrp.production'
|
|
||||||
|
|
||||||
pr_mp_count = fields.Integer('采购申请单数量', compute='_compute_pr_mp_count', store=True)
|
|
||||||
|
|
||||||
@api.depends('state')
|
|
||||||
def _compute_pr_mp_count(self):
|
|
||||||
for item in self:
|
|
||||||
# if item.product_id.product_tmpl_id.single_manufacturing == True and not item.is_remanufacture:
|
|
||||||
# first_order = self.env['mrp.production'].search(
|
|
||||||
# [('origin', '=', item.origin), ('product_id', '=', item.product_id.id)], limit=1, order='id asc')
|
|
||||||
# pr_ids = self.env['purchase.request'].sudo().search([('origin', 'like', first_order.name)])
|
|
||||||
# item.pr_mp_count = len(pr_ids)
|
|
||||||
# else:
|
|
||||||
# pr_ids = self.env['purchase.request'].sudo().search([('origin', 'like', item.name)])
|
|
||||||
# item.pr_mp_count = len(pr_ids)
|
|
||||||
# 由于采购申请合并了所有销售订单行的采购,所以不区分产品
|
|
||||||
mrp_names = self.env['mrp.production'].search([('origin', '=', item.origin)]).mapped('name')
|
|
||||||
pr_ids = self.env['purchase.request'].sudo().search([('origin', 'in', mrp_names)])
|
|
||||||
item.pr_mp_count = len(pr_ids)
|
|
||||||
# pr_ids = self.env['purchase.request'].sudo().search([('origin', 'like', item.name), ('is_subcontract', '!=', 'True')])
|
|
||||||
|
|
||||||
def action_view_pr_mp(self):
|
|
||||||
"""
|
|
||||||
采购请求
|
|
||||||
"""
|
|
||||||
self.ensure_one()
|
|
||||||
# pr_ids = self.env['purchase.request'].sudo().search([('origin', 'like', self.name),('is_subcontract', '!=', True)])
|
|
||||||
# if self.product_id.product_tmpl_id.single_manufacturing == True and not self.is_remanufacture:
|
|
||||||
# first_order = self.env['mrp.production'].search(
|
|
||||||
# [('origin', '=', self.origin), ('product_id', '=', self.product_id.id)], limit=1, order='id asc')
|
|
||||||
# pr_ids = self.env['purchase.request'].sudo().search([('origin', 'like', first_order.name)])
|
|
||||||
# else:
|
|
||||||
# pr_ids = self.env['purchase.request'].sudo().search([('origin', 'like', self.name)])
|
|
||||||
# 由于采购申请合并了所有销售订单行的采购,所以不区分产品
|
|
||||||
mrp_names = self.env['mrp.production'].search([('origin', '=', self.origin)]).mapped('name')
|
|
||||||
pr_ids = self.env['purchase.request'].sudo().search([('origin', 'in', mrp_names)])
|
|
||||||
|
|
||||||
|
|
||||||
action = {
|
|
||||||
'res_model': 'purchase.request',
|
|
||||||
'type': 'ir.actions.act_window',
|
|
||||||
}
|
|
||||||
if len(pr_ids) == 1:
|
|
||||||
action.update({
|
|
||||||
'view_mode': 'form',
|
|
||||||
'res_id': pr_ids[0].id,
|
|
||||||
})
|
|
||||||
else:
|
|
||||||
action.update({
|
|
||||||
'name': _("从 %s生成采购请求单", self.name),
|
|
||||||
'domain': [('id', 'in', pr_ids)],
|
|
||||||
'view_mode': 'tree,form',
|
|
||||||
})
|
|
||||||
return action
|
|
||||||
@@ -1,19 +0,0 @@
|
|||||||
from odoo import models, fields
|
|
||||||
|
|
||||||
|
|
||||||
class ProductTemplate(models.Model):
|
|
||||||
_inherit = 'product.template'
|
|
||||||
|
|
||||||
purchase_request_id = fields.Many2one('purchase.request', string='采购申请')
|
|
||||||
|
|
||||||
def no_bom_product_create(self, product_id, item, order_id, route_type, i, finish_product):
|
|
||||||
""" 创建坯料时,复制采购申请 """
|
|
||||||
template_id = super(ProductTemplate, self).no_bom_product_create(product_id, item, order_id, route_type, i,
|
|
||||||
finish_product)
|
|
||||||
template_id.purchase_request = product_id.purchase_request
|
|
||||||
return template_id
|
|
||||||
|
|
||||||
def copy_template(self, product_template_id):
|
|
||||||
""" 复制成品模板时,复制采购申请 """
|
|
||||||
super(ProductTemplate, self).copy_template(product_template_id)
|
|
||||||
self.purchase_request = product_template_id.purchase_request
|
|
||||||
@@ -1,56 +0,0 @@
|
|||||||
from odoo import api, fields, models, _
|
|
||||||
from odoo.tools import float_compare
|
|
||||||
|
|
||||||
|
|
||||||
class PurchaseOrder(models.Model):
|
|
||||||
_inherit = 'purchase.order'
|
|
||||||
|
|
||||||
state = fields.Selection([
|
|
||||||
('draft', '询价'),
|
|
||||||
('sent', '发送询价'),
|
|
||||||
('to approve', '待批准'),
|
|
||||||
("approved", "已批准"),
|
|
||||||
('purchase', '采购订单'),
|
|
||||||
('done', '完成'),
|
|
||||||
('cancel', '取消'),
|
|
||||||
('rejected', '已驳回')
|
|
||||||
], string='Status', readonly=True, index=True, copy=False, default='draft', tracking=True)
|
|
||||||
|
|
||||||
|
|
||||||
def button_confirm(self):
|
|
||||||
res = super(PurchaseOrder, self).button_confirm()
|
|
||||||
# 取消反向调拨单
|
|
||||||
reverse_move_ids = self.env['stock.move'].search([
|
|
||||||
('origin', '=', self.name),
|
|
||||||
('purchase_line_id', '=', False),
|
|
||||||
('state', '!=', 'done')
|
|
||||||
])
|
|
||||||
if reverse_move_ids:
|
|
||||||
reverse_move_ids.picking_id.action_cancel()
|
|
||||||
return res
|
|
||||||
|
|
||||||
def button_cancel(self):
|
|
||||||
"""
|
|
||||||
将取消的采购订单关联的库存移动撤销
|
|
||||||
"""
|
|
||||||
move_ids = self.order_line.move_dest_ids.filtered(lambda move: move.state != 'done' and not move.scrapped)
|
|
||||||
res =super(PurchaseOrder, self).button_cancel()
|
|
||||||
if move_ids.mapped('created_purchase_request_line_id'):
|
|
||||||
move_ids.write({'state': 'waiting', 'is_done': False})
|
|
||||||
return res
|
|
||||||
|
|
||||||
def write(self, vals):
|
|
||||||
res = super(PurchaseOrder, self).write(vals)
|
|
||||||
if 'state' in vals and vals['state'] == 'purchase':
|
|
||||||
purchase_request = self.order_line.purchase_request_lines.request_id
|
|
||||||
if purchase_request:
|
|
||||||
finished = True
|
|
||||||
# 判断该采购申请所有明细行是否都完成
|
|
||||||
for purchase_request_line in purchase_request.line_ids:
|
|
||||||
finished_qty = sum(purchase_request_line.purchase_lines.filtered(lambda line: line.state == 'purchase').mapped('product_qty'))
|
|
||||||
if float_compare(finished_qty ,purchase_request_line.product_qty, precision_rounding=purchase_request_line.product_id.uom_id.rounding) < 0:
|
|
||||||
finished = False
|
|
||||||
break
|
|
||||||
if finished:
|
|
||||||
purchase_request.button_done()
|
|
||||||
return res
|
|
||||||
@@ -1,199 +0,0 @@
|
|||||||
import re
|
|
||||||
import ast
|
|
||||||
from odoo import models, fields, api, _
|
|
||||||
from itertools import groupby
|
|
||||||
from odoo.tools import float_compare
|
|
||||||
|
|
||||||
|
|
||||||
class PurchaseRequest(models.Model):
|
|
||||||
_inherit = 'purchase.request'
|
|
||||||
_description = '采购申请'
|
|
||||||
|
|
||||||
# 为state添加取消状态
|
|
||||||
state = fields.Selection(
|
|
||||||
selection_add=[('cancel', '已取消')],
|
|
||||||
ondelete={'cancel': 'set default'} # 添加 ondelete 策略
|
|
||||||
)
|
|
||||||
|
|
||||||
rule_new_add = fields.Boolean('采购请求为规则创建', default=False, compute='_compute_state', store=True)
|
|
||||||
|
|
||||||
@api.depends('state')
|
|
||||||
def _compute_state(self):
|
|
||||||
for pr in self:
|
|
||||||
if pr.state != 'draft' and pr.rule_new_add:
|
|
||||||
pr.rule_new_add = False
|
|
||||||
|
|
||||||
def action_view_purchase_order(self):
|
|
||||||
action = super(PurchaseRequest, self).action_view_purchase_order()
|
|
||||||
origin_context = ast.literal_eval(action['context'])
|
|
||||||
if 'search_default_draft' in origin_context:
|
|
||||||
origin_context.pop('search_default_draft')
|
|
||||||
action['context'] = origin_context
|
|
||||||
return action
|
|
||||||
|
|
||||||
def button_done(self):
|
|
||||||
product_qty_map = {key: sum(line.product_qty for line in group) for key, group in
|
|
||||||
groupby(self.line_ids, key=lambda x: x.product_id.id)}
|
|
||||||
lines = self.mapped("line_ids.purchase_lines.order_id")
|
|
||||||
# 采购单产品和数量
|
|
||||||
product_summary = {}
|
|
||||||
product_rounding = {}
|
|
||||||
if lines:
|
|
||||||
for line in lines:
|
|
||||||
for line_item in line.order_line:
|
|
||||||
product_id = line_item.product_id.id
|
|
||||||
qty = line_item.product_qty
|
|
||||||
product_rounding[product_id] = line_item.product_id.uom_id.rounding
|
|
||||||
if product_id in product_summary:
|
|
||||||
product_summary[product_id] += qty
|
|
||||||
else:
|
|
||||||
product_summary[product_id] = qty
|
|
||||||
|
|
||||||
# 校验产品数量
|
|
||||||
discrepancies = []
|
|
||||||
for product_id, qty in product_qty_map.items():
|
|
||||||
if product_id in product_summary:
|
|
||||||
if float_compare(product_summary[product_id], qty, precision_rounding=product_rounding[product_id]) < 0:
|
|
||||||
discrepancies.append((product_id, qty, product_summary[product_id]))
|
|
||||||
else:
|
|
||||||
discrepancies.append((product_id, qty, 0))
|
|
||||||
|
|
||||||
if discrepancies:
|
|
||||||
# 弹出提示框
|
|
||||||
message = "产品数量不一致:\n"
|
|
||||||
for product_id, required_qty, order_qty in discrepancies:
|
|
||||||
product_name = self.env['product.product'].browse(product_id).display_name # 获取产品名称
|
|
||||||
message += f"产品 {product_name},需求数量 {required_qty},关联采购订单数量 {order_qty}(含询价状态)\n"
|
|
||||||
# 添加确认框
|
|
||||||
message += "确认关闭?"
|
|
||||||
return {
|
|
||||||
'name': _('采购申请'),
|
|
||||||
'type': 'ir.actions.act_window',
|
|
||||||
'views': [(self.env.ref(
|
|
||||||
'jikimo_purchase_request.purchase_request_wizard_wizard_form_view').id,
|
|
||||||
'form')],
|
|
||||||
'res_model': 'purchase.request.wizard',
|
|
||||||
'target': 'new',
|
|
||||||
'context': {
|
|
||||||
'default_purchase_request_id': self.id,
|
|
||||||
'default_message': message,
|
|
||||||
}}
|
|
||||||
return super(PurchaseRequest, self).button_done()
|
|
||||||
|
|
||||||
|
|
||||||
class PurchaseRequestLine(models.Model):
|
|
||||||
_inherit = 'purchase.request.line'
|
|
||||||
_description = '采购申请明细'
|
|
||||||
|
|
||||||
origin = fields.Char(string="Source Document")
|
|
||||||
|
|
||||||
part_number = fields.Char('零件图号', store=True, compute='_compute_part_number')
|
|
||||||
part_name = fields.Char('零件名称', store=True, compute='_compute_part_number')
|
|
||||||
related_product = fields.Many2one('product.product', string='关联产品',
|
|
||||||
help='经此产品工艺加工成的成品')
|
|
||||||
|
|
||||||
supply_method = fields.Selection([
|
|
||||||
('automation', "自动化产线加工"),
|
|
||||||
('manual', "人工线下加工"),
|
|
||||||
('purchase', "外购"),
|
|
||||||
('outsourcing', "委外加工"),
|
|
||||||
], string='供货方式', compute='_compute_supply_method', store=True)
|
|
||||||
|
|
||||||
purchase_request_count = fields.Integer(string='采购申请数量', compute='_compute_purchase_request_count',
|
|
||||||
readonly=True)
|
|
||||||
purchase_count = fields.Integer(string="采购订单数量", compute="_compute_purchase_count", readonly=True)
|
|
||||||
|
|
||||||
@api.depends("purchase_lines")
|
|
||||||
def _compute_purchase_count(self):
|
|
||||||
for rec in self:
|
|
||||||
rec.purchase_count = len(rec.mapped("purchase_lines.order_id"))
|
|
||||||
|
|
||||||
@api.depends('request_id')
|
|
||||||
def _compute_purchase_request_count(self):
|
|
||||||
for order in self:
|
|
||||||
order.purchase_request_count = len(order.request_id)
|
|
||||||
|
|
||||||
@api.depends('origin')
|
|
||||||
def _compute_supply_method(self):
|
|
||||||
for prl in self:
|
|
||||||
order_ids = []
|
|
||||||
if not prl.origin:
|
|
||||||
continue
|
|
||||||
origin = [origin.replace(' ', '') for origin in prl.origin.split(',')]
|
|
||||||
if 'S' in prl.origin:
|
|
||||||
# 原单据是销售订单
|
|
||||||
order_ids = self.env['sale.order'].sudo().search([('name', 'in', origin)]).ids
|
|
||||||
elif 'MO' in prl.origin:
|
|
||||||
# 原单据是制造订单
|
|
||||||
mp_ids = self.env['mrp.production'].sudo().search([('name', 'in', origin)])
|
|
||||||
order_ids = [mp_id.sale_order_id.id for mp_id in mp_ids] if mp_ids else []
|
|
||||||
elif 'WH' in prl.origin:
|
|
||||||
# 原单据是调拨单
|
|
||||||
sp_ids = self.env['stock.picking'].sudo().search([('name', 'in', origin)])
|
|
||||||
order_ids = [sp_id.sale_order_id.id for sp_id in sp_ids] if sp_ids else []
|
|
||||||
order_line = self.env['sale.order.line'].sudo().search(
|
|
||||||
[('product_id', '=', prl.product_id.id), ('order_id', 'in', order_ids)])
|
|
||||||
if order_line:
|
|
||||||
prl.supply_method = order_line[0].supply_method
|
|
||||||
else:
|
|
||||||
prl.supply_method = None
|
|
||||||
|
|
||||||
@api.depends('product_id')
|
|
||||||
def _compute_part_number(self):
|
|
||||||
for record in self:
|
|
||||||
if record.part_number and record.part_name:
|
|
||||||
continue
|
|
||||||
if record.product_id.categ_id.name == '坯料':
|
|
||||||
product_name = ''
|
|
||||||
match = re.search(r'(S\d{5}-\d+)', record.product_id.name)
|
|
||||||
# 如果匹配成功,提取结果
|
|
||||||
if match:
|
|
||||||
product_name = match.group(0)
|
|
||||||
sale_order_name = ''
|
|
||||||
match_sale = re.search(r'S(\d+)', record.product_id.name)
|
|
||||||
if match_sale:
|
|
||||||
sale_order_name = match_sale.group(0)
|
|
||||||
sale_order = self.env['sale.order'].sudo().search(
|
|
||||||
[('name', '=', sale_order_name)])
|
|
||||||
if sale_order:
|
|
||||||
filtered_order_line = sale_order.order_line.filtered(
|
|
||||||
lambda order_line: re.search(f'{product_name}$', order_line.product_id.name)
|
|
||||||
)
|
|
||||||
record.part_number = filtered_order_line.product_id.part_number
|
|
||||||
record.part_name = filtered_order_line.product_id.part_name
|
|
||||||
else:
|
|
||||||
record.part_number = record.product_id.part_number
|
|
||||||
record.part_name = record.product_id.part_name
|
|
||||||
|
|
||||||
def _compute_qty_to_buy(self):
|
|
||||||
for pr in self:
|
|
||||||
qty_to_buy = sum(pr.mapped("product_qty"))
|
|
||||||
if pr.purchase_count > 0:
|
|
||||||
qty_to_buy -= sum(pr.mapped("purchase_lines").filtered(lambda po: po.state != 'cancel').mapped(
|
|
||||||
"product_qty"))
|
|
||||||
pr.qty_to_buy = qty_to_buy > 0.0
|
|
||||||
pr.pending_qty_to_receive = qty_to_buy
|
|
||||||
|
|
||||||
def action_view_purchase_request(self):
|
|
||||||
action = self.env["ir.actions.actions"]._for_xml_id("purchase_request.purchase_request_form_action")
|
|
||||||
action.update({
|
|
||||||
'res_id': self.request_id.id,
|
|
||||||
'views': [[False, 'form']],
|
|
||||||
})
|
|
||||||
return action
|
|
||||||
|
|
||||||
def action_view_purchase_order(self):
|
|
||||||
action = self.env["ir.actions.actions"]._for_xml_id("purchase.purchase_rfq")
|
|
||||||
lines = self.mapped("purchase_lines.order_id")
|
|
||||||
if len(lines) > 1:
|
|
||||||
action["domain"] = [("id", "in", lines.ids)]
|
|
||||||
elif lines:
|
|
||||||
action["views"] = [
|
|
||||||
(self.env.ref("purchase.purchase_order_form").id, "form")
|
|
||||||
]
|
|
||||||
action["res_id"] = lines.id
|
|
||||||
origin_context = ast.literal_eval(action['context'])
|
|
||||||
if 'search_default_draft' in origin_context:
|
|
||||||
origin_context.pop('search_default_draft')
|
|
||||||
action['context'] = origin_context
|
|
||||||
return action
|
|
||||||
@@ -1,50 +0,0 @@
|
|||||||
from odoo import fields, models, api, _
|
|
||||||
|
|
||||||
|
|
||||||
class StatusChange(models.Model):
|
|
||||||
_inherit = 'sale.order'
|
|
||||||
|
|
||||||
# def action_confirm(self):
|
|
||||||
# res = super(StatusChange, self).action_confirm()
|
|
||||||
# # 采购申请自动确认
|
|
||||||
# pr_ids = self.env["purchase.request"].sudo().search(
|
|
||||||
# [('origin', 'like', self.name), ('rule_new_add', '=', True)])
|
|
||||||
# if pr_ids:
|
|
||||||
# pr_ids.write({'need_validation': False})
|
|
||||||
# pr_ids.write({"state": "approved"})
|
|
||||||
# return res
|
|
||||||
|
|
||||||
purchase_request_purchase_order_count = fields.Integer('采购申请单数量', compute='_compute_purchase_request_count',
|
|
||||||
store=True)
|
|
||||||
|
|
||||||
@api.depends('state')
|
|
||||||
def _compute_purchase_request_count(self):
|
|
||||||
for so in self:
|
|
||||||
pr_ids = self.env['purchase.request'].sudo().search([('origin', 'like', so.name)])
|
|
||||||
if pr_ids:
|
|
||||||
so.purchase_request_purchase_order_count = len(pr_ids)
|
|
||||||
else:
|
|
||||||
so.purchase_request_purchase_order_count = 0
|
|
||||||
|
|
||||||
def action_view_purchase_request_purchase_orders(self):
|
|
||||||
"""
|
|
||||||
采购请求
|
|
||||||
"""
|
|
||||||
self.ensure_one()
|
|
||||||
pr_ids = self.env['purchase.request'].sudo().search([('origin', 'like', self.name)])
|
|
||||||
action = {
|
|
||||||
'res_model': 'purchase.request',
|
|
||||||
'type': 'ir.actions.act_window',
|
|
||||||
}
|
|
||||||
if len(pr_ids) == 1:
|
|
||||||
action.update({
|
|
||||||
'view_mode': 'form',
|
|
||||||
'res_id': pr_ids[0].id,
|
|
||||||
})
|
|
||||||
else:
|
|
||||||
action.update({
|
|
||||||
'name': _("从 %s生成采购请求单", self.name),
|
|
||||||
'domain': [('id', 'in', pr_ids)],
|
|
||||||
'view_mode': 'tree,form',
|
|
||||||
})
|
|
||||||
return action
|
|
||||||
@@ -1,47 +0,0 @@
|
|||||||
from odoo import fields, api, models, _
|
|
||||||
|
|
||||||
|
|
||||||
class StockPicking(models.Model):
|
|
||||||
_inherit = "stock.picking"
|
|
||||||
|
|
||||||
purchase_request_count = fields.Integer('采购订单数量', compute='_compute_purchase_request')
|
|
||||||
|
|
||||||
@api.depends('name')
|
|
||||||
def _compute_purchase_request(self):
|
|
||||||
for record in self:
|
|
||||||
purchase_request_ids = self.env['purchase.request'].search([('origin', '=', record.name)])
|
|
||||||
record.purchase_request_count = len(purchase_request_ids)
|
|
||||||
|
|
||||||
def action_view_purchase_request(self):
|
|
||||||
self.ensure_one()
|
|
||||||
|
|
||||||
purchase_request_ids = self.env['purchase.request'].search([('origin', '=', self.name)])
|
|
||||||
|
|
||||||
action = {
|
|
||||||
'res_model': 'purchase.request',
|
|
||||||
'type': 'ir.actions.act_window',
|
|
||||||
}
|
|
||||||
if len(purchase_request_ids) == 1:
|
|
||||||
action.update({
|
|
||||||
'view_mode': 'form',
|
|
||||||
'res_id': purchase_request_ids[0].id,
|
|
||||||
})
|
|
||||||
else:
|
|
||||||
action.update({
|
|
||||||
'name': _("从 %s生成采购请求单", self.name),
|
|
||||||
'domain': [('id', 'in', purchase_request_ids.ids)],
|
|
||||||
'view_mode': 'tree,form',
|
|
||||||
})
|
|
||||||
return action
|
|
||||||
|
|
||||||
def _action_done(self):
|
|
||||||
res = super(StockPicking, self)._action_done()
|
|
||||||
# 将新产生的backorder对应上原来的采购申请明细行
|
|
||||||
backorder_ids = self.backorder_ids
|
|
||||||
if backorder_ids:
|
|
||||||
purchase_request_lines = self.move_ids.move_orig_ids.purchase_line_id.purchase_request_lines
|
|
||||||
if purchase_request_lines:
|
|
||||||
purchase_request_lines.move_dest_ids = [
|
|
||||||
(4, x.id) for x in backorder_ids.move_ids if x.product_id.id == purchase_request_lines.product_id.id
|
|
||||||
]
|
|
||||||
return res
|
|
||||||
@@ -1,82 +0,0 @@
|
|||||||
from odoo import api, fields, models
|
|
||||||
from collections import defaultdict
|
|
||||||
|
|
||||||
|
|
||||||
class StockRule(models.Model):
|
|
||||||
_inherit = "stock.rule"
|
|
||||||
|
|
||||||
def create_purchase_request(self, procurement_group):
|
|
||||||
"""
|
|
||||||
Create a purchase request containing procurement order product.
|
|
||||||
"""
|
|
||||||
procurement = procurement_group[0]
|
|
||||||
rule = procurement_group[1]
|
|
||||||
purchase_request_model = self.env["purchase.request"]
|
|
||||||
purchase_request_line_model = self.env["purchase.request.line"]
|
|
||||||
cache = {}
|
|
||||||
pr = self.env["purchase.request"]
|
|
||||||
domain = rule._make_pr_get_domain(procurement.values)
|
|
||||||
if domain in cache:
|
|
||||||
pr = cache[domain]
|
|
||||||
elif domain:
|
|
||||||
pr = self.env["purchase.request"].search([dom for dom in domain])
|
|
||||||
pr = pr[0] if pr else False
|
|
||||||
cache[domain] = pr
|
|
||||||
if not pr:
|
|
||||||
request_data = rule._prepare_purchase_request(
|
|
||||||
procurement.origin, procurement.values
|
|
||||||
)
|
|
||||||
request_data.update({'rule_new_add': True})
|
|
||||||
pr = purchase_request_model.create(request_data)
|
|
||||||
cache[domain] = pr
|
|
||||||
elif (
|
|
||||||
not pr.origin
|
|
||||||
or procurement.origin not in pr.origin.split(", ")
|
|
||||||
and procurement.origin != "/"
|
|
||||||
):
|
|
||||||
if pr.origin:
|
|
||||||
if procurement.origin:
|
|
||||||
pr.write({"origin": pr.origin + ", " + procurement.origin})
|
|
||||||
else:
|
|
||||||
pr.write({"origin": procurement.origin})
|
|
||||||
# Create Line
|
|
||||||
request_line_data = rule._prepare_purchase_request_line(pr, procurement)
|
|
||||||
request_line_data.update({'origin': procurement.origin})
|
|
||||||
purchase_request_line_model.create(request_line_data)
|
|
||||||
|
|
||||||
def _run_buy(self, procurements):
|
|
||||||
# 如果补货组相同,并且产品相同,则合并
|
|
||||||
procurements_dict = defaultdict()
|
|
||||||
for procurement, rule in procurements:
|
|
||||||
if (procurement.product_id.id, procurement.values['group_id'], rule.id) not in procurements_dict:
|
|
||||||
procurements_dict[(procurement.product_id.id, procurement.values['group_id'], rule.id)] = {
|
|
||||||
'product_id': procurement.product_id,
|
|
||||||
'product_qty': procurement.product_qty,
|
|
||||||
'product_uom': procurement.product_uom,
|
|
||||||
'location_id': procurement.location_id,
|
|
||||||
'name': procurement.name,
|
|
||||||
'origin': procurement.origin,
|
|
||||||
'company_id': procurement.company_id,
|
|
||||||
'values': procurement.values,
|
|
||||||
'rule': rule
|
|
||||||
}
|
|
||||||
else:
|
|
||||||
procurements_dict[(procurement.product_id.id, procurement.values['group_id'], rule.id)]['product_qty'] += procurement.product_qty
|
|
||||||
procurements_dict[(procurement.product_id.id, procurement.values['group_id'], rule.id)]['values']['move_dest_ids'] |= procurement.values['move_dest_ids']
|
|
||||||
new_procurements = []
|
|
||||||
for k, p in procurements_dict.items():
|
|
||||||
new_procurements.append((
|
|
||||||
self.env['procurement.group'].Procurement(
|
|
||||||
product_id=p['product_id'],
|
|
||||||
product_qty=p['product_qty'],
|
|
||||||
product_uom=p['product_uom'],
|
|
||||||
location_id=p['location_id'],
|
|
||||||
name=p['name'],
|
|
||||||
origin=p['origin'],
|
|
||||||
company_id=p['company_id'],
|
|
||||||
values=p['values']
|
|
||||||
), p['rule'])
|
|
||||||
)
|
|
||||||
|
|
||||||
res = super(StockRule, self)._run_buy(new_procurements)
|
|
||||||
return res
|
|
||||||
@@ -1,2 +0,0 @@
|
|||||||
id,name,model_id:id,group_id:id,perm_read,perm_write,perm_create,perm_unlink
|
|
||||||
access_purchase_request_wizard_group_user,purchase.request.wizard,model_purchase_request_wizard,base.group_user,1,1,1,1
|
|
||||||
|
@@ -1,3 +0,0 @@
|
|||||||
th[data-name=keep_description] {
|
|
||||||
min-width: 220px;
|
|
||||||
}
|
|
||||||
@@ -1,21 +0,0 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8" ?>
|
|
||||||
<odoo>
|
|
||||||
<record id="mrp_production_inherited_form_purchase_request" model="ir.ui.view">
|
|
||||||
<field name="name">mrp.production.inherited.form.purchase.request</field>
|
|
||||||
<field name="model">mrp.production</field>
|
|
||||||
<field name="inherit_id" ref="mrp.mrp_production_form_view"/>
|
|
||||||
<field name="arch" type="xml">
|
|
||||||
<xpath expr="//button[@name='action_view_mo_delivery']" position="before">
|
|
||||||
<button class="oe_stat_button" name="action_view_pr_mp" type="object" icon="fa-credit-card"
|
|
||||||
attrs="{'invisible': [('pr_mp_count', '=', 0)]}">
|
|
||||||
<div class="o_field_widget o_stat_info">
|
|
||||||
<span class="o_stat_value">
|
|
||||||
<field name="pr_mp_count"/>
|
|
||||||
</span>
|
|
||||||
<span class="o_stat_text">采购申请</span>
|
|
||||||
</div>
|
|
||||||
</button>
|
|
||||||
</xpath>
|
|
||||||
</field>
|
|
||||||
</record>
|
|
||||||
</odoo>
|
|
||||||
@@ -1,22 +0,0 @@
|
|||||||
<odoo>
|
|
||||||
<record id="purchase_request_line_form_sf" model="ir.ui.view">
|
|
||||||
<field name="name">purchase.request.line.sf.form</field>
|
|
||||||
<field name="model">purchase.request.line</field>
|
|
||||||
<field name="inherit_id" ref="purchase_request.purchase_request_line_form"/>
|
|
||||||
<field name="arch" type="xml">
|
|
||||||
<xpath expr="//h1" position="before">
|
|
||||||
<div class="oe_button_box" name="button_box">
|
|
||||||
<button type="object" name="action_view_purchase_request" class="oe_stat_button"
|
|
||||||
icon="fa-file">
|
|
||||||
<field name="purchase_request_count" widget="statinfo" string="采购申请"/>
|
|
||||||
</button>
|
|
||||||
<button type="object" name="action_view_purchase_order" class="oe_stat_button"
|
|
||||||
attrs="{'invisible': [('purchase_count', '=', 0)]}" icon="fa-shopping-cart">
|
|
||||||
<field name="purchase_count" widget="statinfo" string="采购订单"/>
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
</xpath>
|
|
||||||
|
|
||||||
</field>
|
|
||||||
</record>
|
|
||||||
</odoo>
|
|
||||||
@@ -1,21 +0,0 @@
|
|||||||
<?xml version="1.0" encoding="utf-8"?>
|
|
||||||
<odoo>
|
|
||||||
<data>
|
|
||||||
<record id="menu_purhcase_request" model="ir.ui.menu">
|
|
||||||
<field name="name">采购申请</field>
|
|
||||||
<field name="parent_id" ref="purchase.menu_purchase_root" />
|
|
||||||
<field name="sequence">2</field>
|
|
||||||
</record>
|
|
||||||
|
|
||||||
<record id="purchase_request.menu_purchase_request_pro_mgt" model="ir.ui.menu">
|
|
||||||
<field name="sequence">1</field>
|
|
||||||
<field name="parent_id" ref="jikimo_purchase_request.menu_purhcase_request"/>
|
|
||||||
</record>
|
|
||||||
|
|
||||||
<record id="purchase_request.menu_purchase_request_line" model="ir.ui.menu">
|
|
||||||
<field name="sequence">10</field>
|
|
||||||
<field name="parent_id" ref="jikimo_purchase_request.menu_purhcase_request"/>
|
|
||||||
</record>
|
|
||||||
</data>
|
|
||||||
</odoo>
|
|
||||||
|
|
||||||
@@ -1,91 +0,0 @@
|
|||||||
<odoo>
|
|
||||||
<record id="view_purchase_request_form_sf" model="ir.ui.view">
|
|
||||||
<field name="name">purchase.request.sf.form</field>
|
|
||||||
<field name="model">purchase.request</field>
|
|
||||||
<field name="inherit_id" ref="purchase_request.view_purchase_request_form"/>
|
|
||||||
<field name="arch" type="xml">
|
|
||||||
<xpath expr="//button[@name='button_draft']" position="attributes">
|
|
||||||
<attribute name="string">重置草稿</attribute>
|
|
||||||
</xpath>
|
|
||||||
<xpath expr="//field[@name='line_ids']//field[@name='purchased_qty']" position="after">
|
|
||||||
<field name="supply_method"/>
|
|
||||||
</xpath>
|
|
||||||
<xpath expr="//field[@name='line_ids']//field[@name='name']" position="replace">
|
|
||||||
<field name="related_product"/>
|
|
||||||
<field name="part_number"/>
|
|
||||||
<field name="part_name"/>
|
|
||||||
</xpath>
|
|
||||||
<xpath expr="//button[@name='button_done']" position="attributes">
|
|
||||||
<attribute name="class"/>
|
|
||||||
</xpath>
|
|
||||||
<xpath expr="//button[@name='button_in_progress']" position="attributes">
|
|
||||||
<attribute name="invisible">1</attribute>
|
|
||||||
</xpath>
|
|
||||||
<xpath expr="//button[@name='%(purchase_request.action_purchase_request_line_make_purchase_order)d']" position="attributes">
|
|
||||||
<attribute name="class">oe_highlight</attribute>
|
|
||||||
</xpath>
|
|
||||||
</field>
|
|
||||||
</record>
|
|
||||||
|
|
||||||
<record id="view_purchase_request_tree_sf" model="ir.ui.view">
|
|
||||||
<field name="name">purchase.request.sf.tree</field>
|
|
||||||
<field name="model">purchase.request</field>
|
|
||||||
<field name="inherit_id" ref="purchase_request.view_purchase_request_form"/>
|
|
||||||
<field name="arch" type="xml">
|
|
||||||
<xpath expr="//field[@name='activity_ids']" position="attributes">
|
|
||||||
<attribute name="optional">hide</attribute>
|
|
||||||
</xpath>
|
|
||||||
</field>
|
|
||||||
</record>
|
|
||||||
|
|
||||||
<record id="view_purchase_request_line_tree_sf" model="ir.ui.view">
|
|
||||||
<field name="name">purchase.request.line.sf.tree</field>
|
|
||||||
<field name="model">purchase.request.line</field>
|
|
||||||
<field name="inherit_id" ref="purchase_request.purchase_request_line_tree"/>
|
|
||||||
<field name="arch" type="xml">
|
|
||||||
<xpath expr="//field[@name='requested_by']" position="replace">
|
|
||||||
<field name="supply_method"/>
|
|
||||||
</xpath>
|
|
||||||
<xpath expr="//field[@name='assigned_to']" position="attributes">
|
|
||||||
<attribute name="invisible">True</attribute>
|
|
||||||
</xpath>
|
|
||||||
<xpath expr="//field[@name='name']" position="attributes">
|
|
||||||
<attribute name="invisible">True</attribute>
|
|
||||||
</xpath>
|
|
||||||
<xpath expr="//field[@name='supplier_id']" position="after">
|
|
||||||
<field name="requested_by" widget="many2one_avatar_user"/>
|
|
||||||
<field name="assigned_to" widget="many2one_avatar_user" invisible="1"/>
|
|
||||||
</xpath>
|
|
||||||
<xpath expr="//field[@name='purchased_qty']" position="attributes">
|
|
||||||
<attribute name="string">采购数量</attribute>
|
|
||||||
</xpath>
|
|
||||||
<xpath expr="//field[@name='purchase_state']" position="attributes">
|
|
||||||
<attribute name="string">订单状态</attribute>
|
|
||||||
</xpath>
|
|
||||||
<xpath expr="//field[@name='product_id']" position="after">
|
|
||||||
<field name="related_product"/>
|
|
||||||
<field name="part_number"/>
|
|
||||||
<field name="part_name" invisible="1"/>
|
|
||||||
</xpath>
|
|
||||||
</field>
|
|
||||||
</record>
|
|
||||||
|
|
||||||
<record id="view_purchase_request_line_search_sf" model="ir.ui.view">
|
|
||||||
<field name="name">purchase.request.line.sf.search</field>
|
|
||||||
<field name="model">purchase.request.line</field>
|
|
||||||
<field name="inherit_id" ref="purchase_request.purchase_request_line_search"/>
|
|
||||||
<field name="arch" type="xml">
|
|
||||||
<xpath expr="//field[@name='product_id']" position="after">
|
|
||||||
<field name="supply_method"/>
|
|
||||||
<field name="related_product"/>
|
|
||||||
<field name="part_number"/>
|
|
||||||
<field name="part_name"/>
|
|
||||||
</xpath>
|
|
||||||
</field>
|
|
||||||
</record>
|
|
||||||
|
|
||||||
<record model="ir.actions.act_window" id="purchase_request.purchase_request_form_action">
|
|
||||||
<field name="name">Purchase Requests</field>
|
|
||||||
<field name="context"></field>
|
|
||||||
</record>
|
|
||||||
</odoo>
|
|
||||||
@@ -1,19 +0,0 @@
|
|||||||
<odoo>
|
|
||||||
<record id="sale_order_inherited_form_purchase_request_sf" model="ir.ui.view">
|
|
||||||
<field name="name">sale.order.inherited.form.purchase.request</field>
|
|
||||||
<field name="model">sale.order</field>
|
|
||||||
<field name="inherit_id" ref="sale_purchase.sale_order_inherited_form_purchase"/>
|
|
||||||
<field name="arch" type="xml">
|
|
||||||
<xpath expr="//button[@name='action_preview_sale_order']" position="before">
|
|
||||||
<button class="oe_stat_button" name="action_view_purchase_request_purchase_orders" type="object" icon="fa-credit-card"
|
|
||||||
groups='purchase.group_purchase_user'
|
|
||||||
attrs="{'invisible': [('purchase_request_purchase_order_count', '=', 0)]}">
|
|
||||||
<div class="o_field_widget o_stat_info">
|
|
||||||
<span class="o_stat_value"><field name="purchase_request_purchase_order_count"/></span>
|
|
||||||
<span class="o_stat_text">采购申请</span>
|
|
||||||
</div>
|
|
||||||
</button>
|
|
||||||
</xpath>
|
|
||||||
</field>
|
|
||||||
</record>
|
|
||||||
</odoo>
|
|
||||||
@@ -1,21 +0,0 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8" ?>
|
|
||||||
<odoo>
|
|
||||||
<record id="stock_pikcing_inherited_form_jikimo_purchase_request" model="ir.ui.view">
|
|
||||||
<field name="name">stock.pikcing.inherited.form.jikimo.purchase.request</field>
|
|
||||||
<field name="model">stock.picking</field>
|
|
||||||
<field name="inherit_id" ref="stock.view_picking_form"/>
|
|
||||||
<field name="arch" type="xml">
|
|
||||||
<xpath expr="//div[@name='button_box']/button" position="before">
|
|
||||||
<button class="oe_stat_button" name="action_view_purchase_request" type="object" icon="fa-credit-card"
|
|
||||||
attrs="{'invisible': [('purchase_request_count', '=', 0)]}">
|
|
||||||
<div class="o_field_widget o_stat_info">
|
|
||||||
<span class="o_stat_value">
|
|
||||||
<field name="purchase_request_count"/>
|
|
||||||
</span>
|
|
||||||
<span class="o_stat_text">采购申请</span>
|
|
||||||
</div>
|
|
||||||
</button>
|
|
||||||
</xpath>
|
|
||||||
</field>
|
|
||||||
</record>
|
|
||||||
</odoo>
|
|
||||||
@@ -1,4 +0,0 @@
|
|||||||
# License LGPL-3.0 or later (https://www.gnu.org/licenses/lgpl-3.0)
|
|
||||||
|
|
||||||
from . import purchase_request_line_make_purchase_order
|
|
||||||
from . import purchase_request_wizard
|
|
||||||
@@ -1,129 +0,0 @@
|
|||||||
# Copyright 2018-2019 ForgeFlow, S.L.
|
|
||||||
# License LGPL-3.0 or later (https://www.gnu.org/licenses/lgpl-3.0).
|
|
||||||
from datetime import datetime
|
|
||||||
|
|
||||||
from odoo import _, api, fields, models
|
|
||||||
from odoo.exceptions import UserError, ValidationError
|
|
||||||
from odoo.tools import get_lang
|
|
||||||
|
|
||||||
|
|
||||||
class PurchaseRequestLineMakePurchaseOrder(models.TransientModel):
|
|
||||||
_inherit = "purchase.request.line.make.purchase.order"
|
|
||||||
|
|
||||||
def make_purchase_order(self):
|
|
||||||
res = []
|
|
||||||
purchase_obj = self.env["purchase.order"]
|
|
||||||
po_line_obj = self.env["purchase.order.line"]
|
|
||||||
purchase = False
|
|
||||||
|
|
||||||
if len(set([item_id.line_id.supply_method for item_id in self.item_ids])) > 1:
|
|
||||||
raise ValidationError('不同供货方式不可合并创建询价单!')
|
|
||||||
|
|
||||||
for item in self.item_ids:
|
|
||||||
line = item.line_id
|
|
||||||
if item.product_qty <= 0.0:
|
|
||||||
raise UserError(_("Enter a positive quantity."))
|
|
||||||
if self.purchase_order_id:
|
|
||||||
purchase = self.purchase_order_id
|
|
||||||
if not purchase:
|
|
||||||
po_data = self._prepare_purchase_order(
|
|
||||||
line.request_id.picking_type_id,
|
|
||||||
line.request_id.group_id,
|
|
||||||
line.company_id,
|
|
||||||
line.request_id.origin,
|
|
||||||
)
|
|
||||||
# po_data.update({'related_product':line.related_product.id})
|
|
||||||
purchase = purchase_obj.create(po_data)
|
|
||||||
|
|
||||||
# Look for any other PO line in the selected PO with same
|
|
||||||
# product and UoM to sum quantities instead of creating a new
|
|
||||||
# po line
|
|
||||||
domain = self._get_order_line_search_domain(purchase, item)
|
|
||||||
available_po_lines = po_line_obj.search(domain)
|
|
||||||
new_pr_line = True
|
|
||||||
# If Unit of Measure is not set, update from wizard.
|
|
||||||
if not line.product_uom_id:
|
|
||||||
line.product_uom_id = item.product_uom_id
|
|
||||||
# Allocation UoM has to be the same as PR line UoM
|
|
||||||
alloc_uom = line.product_uom_id
|
|
||||||
wizard_uom = item.product_uom_id
|
|
||||||
if available_po_lines and not item.keep_description:
|
|
||||||
new_pr_line = False
|
|
||||||
po_line = available_po_lines[0]
|
|
||||||
po_line.purchase_request_lines = [(4, line.id)]
|
|
||||||
po_line.move_dest_ids |= line.move_dest_ids
|
|
||||||
po_line_product_uom_qty = po_line.product_uom._compute_quantity(
|
|
||||||
po_line.product_uom_qty, alloc_uom
|
|
||||||
)
|
|
||||||
wizard_product_uom_qty = wizard_uom._compute_quantity(
|
|
||||||
item.product_qty, alloc_uom
|
|
||||||
)
|
|
||||||
all_qty = min(po_line_product_uom_qty, wizard_product_uom_qty)
|
|
||||||
self.create_allocation(po_line, line, all_qty, alloc_uom)
|
|
||||||
else:
|
|
||||||
po_line_data = self._prepare_purchase_order_line(purchase, item)
|
|
||||||
if item.keep_description:
|
|
||||||
po_line_data["name"] = item.name
|
|
||||||
if line.related_product:
|
|
||||||
po_line_data.update({'related_product': line.related_product.id})
|
|
||||||
po_line = po_line_obj.create(po_line_data)
|
|
||||||
po_line_product_uom_qty = po_line.product_uom._compute_quantity(
|
|
||||||
po_line.product_uom_qty, alloc_uom
|
|
||||||
)
|
|
||||||
wizard_product_uom_qty = wizard_uom._compute_quantity(
|
|
||||||
item.product_qty, alloc_uom
|
|
||||||
)
|
|
||||||
all_qty = min(po_line_product_uom_qty, wizard_product_uom_qty)
|
|
||||||
self.create_allocation(po_line, line, all_qty, alloc_uom)
|
|
||||||
self._post_process_po_line(item, po_line, new_pr_line)
|
|
||||||
res.append(purchase.id)
|
|
||||||
|
|
||||||
purchase_requests = self.item_ids.mapped("request_id")
|
|
||||||
purchase_requests.button_in_progress()
|
|
||||||
return {
|
|
||||||
"domain": [("id", "in", res)],
|
|
||||||
"name": _("RFQ"),
|
|
||||||
"view_mode": "tree,form",
|
|
||||||
"res_model": "purchase.order",
|
|
||||||
"view_id": False,
|
|
||||||
"context": False,
|
|
||||||
"type": "ir.actions.act_window",
|
|
||||||
}
|
|
||||||
|
|
||||||
def _check_valid_request_line(self, request_line_ids):
|
|
||||||
for line in self.env["purchase.request.line"].browse(request_line_ids):
|
|
||||||
if line.request_id.state not in ["approved", "in_progress"]:
|
|
||||||
raise UserError(
|
|
||||||
_("采购申请 %s 未审批或未进行中")
|
|
||||||
% line.request_id.name
|
|
||||||
)
|
|
||||||
super(PurchaseRequestLineMakePurchaseOrder, self)._check_valid_request_line(request_line_ids)
|
|
||||||
|
|
||||||
@api.model
|
|
||||||
def check_group(self, request_lines):
|
|
||||||
# 去掉合并必须同一采购组的限制
|
|
||||||
pass
|
|
||||||
|
|
||||||
def get_items(self, request_line_ids):
|
|
||||||
request_line_obj = self.env["purchase.request.line"]
|
|
||||||
items = []
|
|
||||||
request_lines = request_line_obj.browse(request_line_ids).filtered(lambda line: line.pending_qty_to_receive > 0)
|
|
||||||
self._check_valid_request_line(request_line_ids)
|
|
||||||
self.check_group(request_lines)
|
|
||||||
for line in request_lines:
|
|
||||||
items.append([0, 0, self._prepare_item(line)])
|
|
||||||
return items
|
|
||||||
|
|
||||||
|
|
||||||
class PurchaseRequestLineMakePurchaseOrderItem(models.TransientModel):
|
|
||||||
_inherit = "purchase.request.line.make.purchase.order.item"
|
|
||||||
|
|
||||||
supply_method = fields.Selection(related='line_id.supply_method', string='供货方式')
|
|
||||||
|
|
||||||
wiz_id = fields.Many2one(
|
|
||||||
comodel_name="purchase.request.line.make.purchase.order",
|
|
||||||
string="Wizard",
|
|
||||||
required=False,
|
|
||||||
ondelete="cascade",
|
|
||||||
readonly=True,
|
|
||||||
)
|
|
||||||
@@ -1,13 +0,0 @@
|
|||||||
<?xml version="1.0" encoding="utf-8" ?>
|
|
||||||
<odoo>
|
|
||||||
<record id="view_purchase_request_line_make_purchase_order_sf" model="ir.ui.view">
|
|
||||||
<field name="name">Purchase Request Line Make Purchase Order sf</field>
|
|
||||||
<field name="model">purchase.request.line.make.purchase.order</field>
|
|
||||||
<field name="inherit_id" ref="purchase_request.view_purchase_request_line_make_purchase_order"/>
|
|
||||||
<field name="arch" type="xml">
|
|
||||||
<xpath expr="//field[@name='item_ids']//field[@name='keep_description']" position="before">
|
|
||||||
<field name="supply_method"/>
|
|
||||||
</xpath>
|
|
||||||
</field>
|
|
||||||
</record>
|
|
||||||
</odoo>
|
|
||||||
@@ -1,12 +0,0 @@
|
|||||||
from odoo import models, fields, api
|
|
||||||
|
|
||||||
|
|
||||||
class PurchaseRequestWizard(models.TransientModel):
|
|
||||||
_name = 'purchase.request.wizard'
|
|
||||||
_description = '采购申请向导'
|
|
||||||
|
|
||||||
purchase_request_id = fields.Many2one('purchase.request', string='采购申请')
|
|
||||||
message = fields.Char(string='提示', readonly=True)
|
|
||||||
|
|
||||||
def confirm(self):
|
|
||||||
return self.purchase_request_id.write({"state": "done"})
|
|
||||||
@@ -1,22 +0,0 @@
|
|||||||
<?xml version="1.0" encoding="utf-8"?>
|
|
||||||
<odoo>
|
|
||||||
<record model="ir.ui.view" id="purchase_request_wizard_wizard_form_view">
|
|
||||||
<field name="name">purchase.request.wizard.form.view</field>
|
|
||||||
<field name="model">purchase.request.wizard</field>
|
|
||||||
<field name="arch" type="xml">
|
|
||||||
<form>
|
|
||||||
<sheet>
|
|
||||||
<div>
|
|
||||||
<div style="white-space: pre-wrap;">
|
|
||||||
<field name="message"/>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<footer>
|
|
||||||
<button string="确认" name="confirm" type="object" class="oe_highlight"/>
|
|
||||||
<button string="取消" class="btn btn-secondary" special="cancel"/>
|
|
||||||
</footer>
|
|
||||||
</sheet>
|
|
||||||
</form>
|
|
||||||
</field>
|
|
||||||
</record>
|
|
||||||
</odoo>
|
|
||||||
@@ -1,3 +0,0 @@
|
|||||||
# -*- coding: utf-8 -*-
|
|
||||||
|
|
||||||
from . import models
|
|
||||||
@@ -1,27 +0,0 @@
|
|||||||
# -*- coding: utf-8 -*-
|
|
||||||
{
|
|
||||||
'name': "机企猫 采购申请审批流程",
|
|
||||||
|
|
||||||
'summary': """
|
|
||||||
采购申请审批流程""",
|
|
||||||
|
|
||||||
'description': """
|
|
||||||
Long description of module's purpose
|
|
||||||
""",
|
|
||||||
|
|
||||||
'author': "My Company",
|
|
||||||
'website': "https://www.yourcompany.com",
|
|
||||||
|
|
||||||
# Categories can be used to filter modules in modules listing
|
|
||||||
# Check https://github.com/odoo/odoo/blob/16.0/odoo/addons/base/data/ir_module_category_data.xml
|
|
||||||
# for the full list
|
|
||||||
'category': 'Uncategorized',
|
|
||||||
'version': '0.1',
|
|
||||||
|
|
||||||
# any module necessary for this one to work correctly
|
|
||||||
'depends': ['purchase_request_tier_validation'],
|
|
||||||
|
|
||||||
# always loaded
|
|
||||||
'data': [
|
|
||||||
],
|
|
||||||
}
|
|
||||||
@@ -1,4 +0,0 @@
|
|||||||
# -*- coding: utf-8 -*-
|
|
||||||
|
|
||||||
from . import models
|
|
||||||
from . import stock_rule
|
|
||||||
@@ -1,30 +0,0 @@
|
|||||||
from odoo import models, fields, api, _
|
|
||||||
from odoo.exceptions import ValidationError
|
|
||||||
import logging
|
|
||||||
|
|
||||||
_logger = logging.getLogger(__name__)
|
|
||||||
|
|
||||||
|
|
||||||
class PurchaseRequest(models.Model):
|
|
||||||
_inherit = 'purchase.request'
|
|
||||||
|
|
||||||
|
|
||||||
def _validate_tier(self, tiers=False):
|
|
||||||
res = super(PurchaseRequest, self)._validate_tier(tiers)
|
|
||||||
|
|
||||||
# 检查是否所有审批都已通过
|
|
||||||
all_approved = all(
|
|
||||||
tier_review.status == 'approved'
|
|
||||||
for tier_review in self.review_ids
|
|
||||||
)
|
|
||||||
|
|
||||||
if self.review_ids and all_approved: # 确保有审批记录
|
|
||||||
self.state = 'approved'
|
|
||||||
|
|
||||||
return res
|
|
||||||
|
|
||||||
@api.model
|
|
||||||
def _get_under_validation_exceptions(self):
|
|
||||||
res = super(PurchaseRequest, self)._get_under_validation_exceptions()
|
|
||||||
res.append("state")
|
|
||||||
return res
|
|
||||||
@@ -1,16 +0,0 @@
|
|||||||
from odoo import models, api
|
|
||||||
|
|
||||||
class StockRule(models.Model):
|
|
||||||
_inherit = 'stock.rule'
|
|
||||||
|
|
||||||
def _run_buy(self, procurements):
|
|
||||||
res = super(StockRule, self)._run_buy(procurements)
|
|
||||||
# 判断是否根据规则生成新的采购申请单据,如果生成则修改状态为 approved
|
|
||||||
origins = list(set([procurement[0].origin for procurement in procurements]))
|
|
||||||
for origin in origins:
|
|
||||||
pr_ids = self.env["purchase.request"].sudo().search(
|
|
||||||
[('origin', 'like', origin), ('rule_new_add', '=', True), ('state', '=', 'draft')])
|
|
||||||
if pr_ids:
|
|
||||||
pr_ids.write({'need_validation': False})
|
|
||||||
pr_ids.write({"state": "approved", 'need_validation': True, 'rule_new_add': False})
|
|
||||||
return res
|
|
||||||
@@ -1,4 +0,0 @@
|
|||||||
# -*- coding: utf-8 -*-
|
|
||||||
|
|
||||||
from . import models
|
|
||||||
from . import wizards
|
|
||||||
@@ -1,37 +0,0 @@
|
|||||||
# -*- coding: utf-8 -*-
|
|
||||||
{
|
|
||||||
'name': "机企猫 采购审批流程",
|
|
||||||
|
|
||||||
'summary': """
|
|
||||||
采购审批流程""",
|
|
||||||
|
|
||||||
'description': """
|
|
||||||
采购审批流程""",
|
|
||||||
|
|
||||||
'author': "My Company",
|
|
||||||
'website': "https://www.yourcompany.com",
|
|
||||||
|
|
||||||
# Categories can be used to filter modules in modules listing
|
|
||||||
# Check https://github.com/odoo/odoo/blob/16.0/odoo/addons/base/data/ir_module_category_data.xml
|
|
||||||
# for the full list
|
|
||||||
'category': 'Uncategorized',
|
|
||||||
'version': '0.1',
|
|
||||||
|
|
||||||
# any module necessary for this one to work correctly
|
|
||||||
'depends': ['purchase', 'purchase_tier_validation', 'documents', 'purchase_request', 'account', 'purchase_order_approved'],
|
|
||||||
|
|
||||||
# always loaded
|
|
||||||
'data': [
|
|
||||||
'views/views.xml',
|
|
||||||
],
|
|
||||||
# only loaded in demonstration mode
|
|
||||||
'demo': [
|
|
||||||
'demo/demo.xml',
|
|
||||||
],
|
|
||||||
|
|
||||||
'assets': {
|
|
||||||
'web.assets_backend': [
|
|
||||||
'jikimo_purchase_tier_validation/static/src/js/ir_model_extend.js',
|
|
||||||
],
|
|
||||||
},
|
|
||||||
}
|
|
||||||
File diff suppressed because it is too large
Load Diff
@@ -1,3 +0,0 @@
|
|||||||
# -*- coding: utf-8 -*-
|
|
||||||
|
|
||||||
from . import models
|
|
||||||
@@ -1,180 +0,0 @@
|
|||||||
from odoo import models, fields, api, _
|
|
||||||
from odoo.exceptions import ValidationError
|
|
||||||
import logging
|
|
||||||
|
|
||||||
_logger = logging.getLogger(__name__)
|
|
||||||
|
|
||||||
|
|
||||||
class jikimo_purchase_tier_validation(models.Model):
|
|
||||||
_name = 'purchase.order'
|
|
||||||
_inherit = ['purchase.order', 'tier.validation']
|
|
||||||
_description = "采购订单"
|
|
||||||
_state_from = ["draft", "to approve", "rejected"]
|
|
||||||
_state_to = ["approved", "purchase"]
|
|
||||||
|
|
||||||
_tier_validation_buttons_xpath = "/form/header/button[@id='draft_confirm'][1]"
|
|
||||||
|
|
||||||
contract_document_id = fields.Many2one('documents.document', string='合同文件')
|
|
||||||
contract_file = fields.Binary(related='contract_document_id.datas', string='合同文件内容')
|
|
||||||
contract_file_name = fields.Char(related='contract_document_id.attachment_id.name', string='文件名')
|
|
||||||
|
|
||||||
# 是否已上传合同文件
|
|
||||||
is_upload_contract_file = fields.Boolean(string='是否已上传合同文件', default=False)
|
|
||||||
|
|
||||||
def button_confirm(self):
|
|
||||||
# for record in self:
|
|
||||||
# if record.need_validation and not record.validation_status == 'validated':
|
|
||||||
# raise ValidationError(_('请先完成审批。'))
|
|
||||||
res = super(jikimo_purchase_tier_validation, self).button_confirm()
|
|
||||||
for record in self:
|
|
||||||
if record.state == 'approved':
|
|
||||||
record.order_line._validate_analytic_distribution()
|
|
||||||
record._add_supplier_to_product()
|
|
||||||
# Deal with double validation process
|
|
||||||
if record._approval_allowed():
|
|
||||||
record.button_approve()
|
|
||||||
if record.partner_id not in record.message_partner_ids:
|
|
||||||
record.message_subscribe([record.partner_id.id])
|
|
||||||
return res
|
|
||||||
|
|
||||||
def request_validation(self):
|
|
||||||
for record in self:
|
|
||||||
# 添加通知消息
|
|
||||||
if hasattr(record, 'message_post'):
|
|
||||||
current_user = self.env.user.name
|
|
||||||
record.message_post(
|
|
||||||
body=f"<strong>{current_user}</strong> 提交审批",
|
|
||||||
message_type='notification',
|
|
||||||
subtype_xmlid='mail.mt_note'
|
|
||||||
)
|
|
||||||
res = super(jikimo_purchase_tier_validation, self).request_validation()
|
|
||||||
self.state = 'to approve'
|
|
||||||
return res
|
|
||||||
|
|
||||||
def restart_validation(self):
|
|
||||||
res = super(jikimo_purchase_tier_validation, self).restart_validation()
|
|
||||||
self.state = 'draft'
|
|
||||||
return res
|
|
||||||
|
|
||||||
def _validate_tier(self, tiers=False):
|
|
||||||
res = super(jikimo_purchase_tier_validation, self)._validate_tier(tiers)
|
|
||||||
|
|
||||||
# 检查是否所有审批都已通过
|
|
||||||
all_approved = all(
|
|
||||||
tier_review.status == 'approved'
|
|
||||||
for tier_review in self.review_ids
|
|
||||||
)
|
|
||||||
|
|
||||||
if self.review_ids and all_approved: # 确保有审批记录
|
|
||||||
self.state = 'approved'
|
|
||||||
|
|
||||||
return res
|
|
||||||
|
|
||||||
@api.model
|
|
||||||
def _get_under_validation_exceptions(self):
|
|
||||||
res = super(jikimo_purchase_tier_validation, self)._get_under_validation_exceptions()
|
|
||||||
res.append("state")
|
|
||||||
return res
|
|
||||||
|
|
||||||
# 上传合同文件
|
|
||||||
def upload_contract_file(self):
|
|
||||||
print('upload_contract_file===========================')
|
|
||||||
# self.ensure_one()
|
|
||||||
# return {
|
|
||||||
# 'name': _('上传合同文件'),
|
|
||||||
# 'type': 'ir.actions.act_window',
|
|
||||||
# 'res_model': 'ir.attachment',
|
|
||||||
# 'view_mode': 'form',
|
|
||||||
# 'view_type': 'form',
|
|
||||||
# 'target': 'new',
|
|
||||||
# 'context': {
|
|
||||||
# 'default_res_model': self._name,
|
|
||||||
# 'default_res_id': self.id,
|
|
||||||
# 'default_type': 'binary',
|
|
||||||
# 'default_mimetype': 'application/pdf,application/msword,application/vnd.openxmlformats-officedocument.wordprocessingml.document,image/jpeg,image/png',
|
|
||||||
# }
|
|
||||||
# }
|
|
||||||
|
|
||||||
self.ensure_one()
|
|
||||||
action = {
|
|
||||||
'type': 'ir.actions.act_window',
|
|
||||||
'name': _('上传合同文件'),
|
|
||||||
'res_model': 'ir.attachment.wizard', # 我们需要创建一个新的向导模型
|
|
||||||
'view_mode': 'form',
|
|
||||||
'target': 'new',
|
|
||||||
'context': {
|
|
||||||
'default_res_model': self._name,
|
|
||||||
'default_res_id': self.id,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return action
|
|
||||||
|
|
||||||
# 删除合同文件
|
|
||||||
def delete_contract_file(self):
|
|
||||||
self.ensure_one()
|
|
||||||
if self.contract_document_id:
|
|
||||||
try:
|
|
||||||
document = self.contract_document_id
|
|
||||||
|
|
||||||
# 清空关联
|
|
||||||
self.write({
|
|
||||||
'contract_document_id': False,
|
|
||||||
'contract_file': False,
|
|
||||||
'contract_file_name': False
|
|
||||||
})
|
|
||||||
|
|
||||||
# 删除文档
|
|
||||||
if document:
|
|
||||||
document.with_context(no_attachment=True).sudo().unlink()
|
|
||||||
|
|
||||||
self.is_upload_contract_file = False
|
|
||||||
|
|
||||||
# 返回视图动作来刷新当前表单
|
|
||||||
return {
|
|
||||||
'type': 'ir.actions.act_window',
|
|
||||||
'res_model': 'purchase.order',
|
|
||||||
'res_id': self.id,
|
|
||||||
'view_mode': 'form',
|
|
||||||
'view_type': 'form',
|
|
||||||
'target': 'current',
|
|
||||||
'flags': {'mode': 'readonly'},
|
|
||||||
}
|
|
||||||
|
|
||||||
except Exception as e:
|
|
||||||
_logger.error('删除合同文件时出错: %s', str(e))
|
|
||||||
return {
|
|
||||||
'type': 'ir.actions.client',
|
|
||||||
'tag': 'display_notification',
|
|
||||||
'params': {
|
|
||||||
'title': _('错误'),
|
|
||||||
'message': _('删除文件时出现错误'),
|
|
||||||
'type': 'danger',
|
|
||||||
'sticky': True,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return {
|
|
||||||
'type': 'ir.actions.client',
|
|
||||||
'tag': 'display_notification',
|
|
||||||
'params': {
|
|
||||||
'title': _('提示'),
|
|
||||||
'message': _('没有需要删除的合同文件'),
|
|
||||||
'type': 'warning',
|
|
||||||
'sticky': False,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
# class jikimo_purchase_request(models.Model):
|
|
||||||
# _inherit = 'purchase.request'
|
|
||||||
# _description = "采购申请"
|
|
||||||
|
|
||||||
|
|
||||||
# class jikimo_account_payment(models.Model):
|
|
||||||
# _inherit = 'account.payment'
|
|
||||||
# _description = "付款单"
|
|
||||||
|
|
||||||
|
|
||||||
# class jikimo_account_move(models.Model):
|
|
||||||
# _inherit = 'account.move'
|
|
||||||
# _description = "发票账单"
|
|
||||||
@@ -1,14 +0,0 @@
|
|||||||
/** @odoo-module **/
|
|
||||||
|
|
||||||
import {registerPatch} from "@mail/model/model_core";
|
|
||||||
|
|
||||||
registerPatch({
|
|
||||||
name: "ir.model.review",
|
|
||||||
fields: {
|
|
||||||
availableWebViews: {
|
|
||||||
compute() {
|
|
||||||
return ["list", "form", "activity"];
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
});
|
|
||||||
@@ -1,32 +0,0 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8" ?>
|
|
||||||
<odoo>
|
|
||||||
<data>
|
|
||||||
<record model="ir.ui.view" id="tier_validation_view_approved_purchase_order_form_inherit">
|
|
||||||
<field name="name">tier_validation_view_approved_purchase_order_form_inherit</field>
|
|
||||||
<field name="model">purchase.order</field>
|
|
||||||
<field name="inherit_id" ref="purchase_order_approved.purchase_order_form"/>
|
|
||||||
<field name="arch" type="xml">
|
|
||||||
<xpath expr="//button[@name='button_release']" position="replace">
|
|
||||||
</xpath>
|
|
||||||
</field>
|
|
||||||
</record>
|
|
||||||
|
|
||||||
<record model="ir.ui.view" id="tier_validation_view_purchase_order_form_inherit">
|
|
||||||
<field name="name">tier_validation_view_purchase_order_form_inherit</field>
|
|
||||||
<field name="model">purchase.order</field>
|
|
||||||
<field name="inherit_id" ref="purchase.purchase_order_form"/>
|
|
||||||
<field name="arch" type="xml">
|
|
||||||
<xpath expr="//header/button[@name='button_approve']" position="replace">
|
|
||||||
</xpath>
|
|
||||||
<xpath expr="//header/button[@name='button_cancel']" position="replace">
|
|
||||||
</xpath>
|
|
||||||
<xpath expr="//header/field[@name='state']" position="replace">
|
|
||||||
<field name="state" widget="statusbar" statusbar_visible="draft,sent,to approve, approved, purchase" readonly="1"/>
|
|
||||||
</xpath>
|
|
||||||
<xpath expr="//header/button[last()]" position="after">
|
|
||||||
<button name="button_cancel" states="draft,to approve,sent,purchase" string="取消" type="object" data-hotkey="x" />
|
|
||||||
</xpath>
|
|
||||||
</field>
|
|
||||||
</record>
|
|
||||||
</data>
|
|
||||||
</odoo>
|
|
||||||
@@ -1 +0,0 @@
|
|||||||
from . import comment_wizard
|
|
||||||
@@ -1,15 +0,0 @@
|
|||||||
from odoo import models, fields
|
|
||||||
|
|
||||||
|
|
||||||
class CommentWizard(models.TransientModel):
|
|
||||||
_inherit = "comment.wizard"
|
|
||||||
|
|
||||||
def add_comment(self):
|
|
||||||
|
|
||||||
rec = self.env[self.res_model].browse(self.res_id)
|
|
||||||
|
|
||||||
self.review_ids = rec.review_ids
|
|
||||||
|
|
||||||
result = super(CommentWizard, self).add_comment()
|
|
||||||
|
|
||||||
return result
|
|
||||||
@@ -1,139 +0,0 @@
|
|||||||
# -*- coding: utf-8 -*-
|
|
||||||
from . import models
|
|
||||||
from . import controllers
|
|
||||||
|
|
||||||
from odoo import api, SUPERUSER_ID
|
|
||||||
|
|
||||||
def _data_install(cr, registry):
|
|
||||||
env = api.Environment(cr, SUPERUSER_ID, {})
|
|
||||||
# 获取所有需要设置的产品模板
|
|
||||||
env.ref('jikimo_sale_multiple_supply_methods.product_template_purchase').product_variant_id.write({'active': False, 'is_bfm': True})
|
|
||||||
env.ref('jikimo_sale_multiple_supply_methods.product_template_manual_processing').product_variant_id.write({'active': False, 'single_manufacturing': True, 'is_bfm': True})
|
|
||||||
env.ref('jikimo_sale_multiple_supply_methods.product_template_default').product_variant_id.write({'active': False, 'is_bfm': True})
|
|
||||||
env.ref('jikimo_sale_multiple_supply_methods.product_template_embryo_customer_provided').product_variant_id.write({'active': False})
|
|
||||||
env.ref('jikimo_sale_multiple_supply_methods.product_template_outsourcing').product_variant_id.write({'active': False, 'is_bfm': True})
|
|
||||||
|
|
||||||
# 为三步制造增加规则
|
|
||||||
warehouse = env['stock.warehouse'].search([('company_id', '=', env.company.id)], limit=1)
|
|
||||||
product_route_id = warehouse.pbm_route_id
|
|
||||||
# 创建规则:原料存货区 -> 制造前, 坯料存货区 -> 制造前, 制造后 -> 坯料存货区, 制造后 -> 成品存货区
|
|
||||||
raw_material_location_id = env['stock.location'].search([('name', '=', '坯料存货区')], limit=1)
|
|
||||||
picking_type_production = warehouse.pbm_type_id
|
|
||||||
picking_type_stock = warehouse.sam_type_id
|
|
||||||
material_location_id = env['stock.location'].search([('name', '=', '原料存货区')], limit=1)
|
|
||||||
# 为mto增加规则
|
|
||||||
mto_route_id = env.ref('stock.route_warehouse0_mto')
|
|
||||||
# 创建规则:原料存货区 -> 外包位置, 坯料存货区 -> 外包位置
|
|
||||||
subcontracting_location_id = env.company.subcontracting_location_id
|
|
||||||
picking_type_subcontracting = warehouse.subcontracting_resupply_type_id
|
|
||||||
# 为补给外包商增加规则
|
|
||||||
resupply_route_id = warehouse.subcontracting_route_id
|
|
||||||
|
|
||||||
rules_data = [
|
|
||||||
{
|
|
||||||
'name': 'WH: 原料存货区 → 制造前',
|
|
||||||
'location_src_id': material_location_id.id,
|
|
||||||
'location_dest_id': warehouse.pbm_loc_id.id,
|
|
||||||
'route_id': product_route_id.id,
|
|
||||||
'picking_type_id': picking_type_production.id,
|
|
||||||
'action': 'pull',
|
|
||||||
'sequence': 20,
|
|
||||||
'warehouse_id': warehouse.id,
|
|
||||||
'procure_method': 'mts_else_mto',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
'name': 'WH: 坯料存货区 → 制造前',
|
|
||||||
'location_src_id': raw_material_location_id.id,
|
|
||||||
'location_dest_id': warehouse.pbm_loc_id.id,
|
|
||||||
'route_id': product_route_id.id,
|
|
||||||
'picking_type_id': picking_type_production.id,
|
|
||||||
'action': 'pull',
|
|
||||||
'sequence': 21,
|
|
||||||
'warehouse_id': warehouse.id,
|
|
||||||
'procure_method': 'mts_else_mto',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
'name': 'WH: 制造后 → 坯料存货区',
|
|
||||||
'location_src_id': warehouse.sam_loc_id.id,
|
|
||||||
'location_dest_id': raw_material_location_id.id,
|
|
||||||
'route_id': product_route_id.id,
|
|
||||||
'picking_type_id': picking_type_stock.id,
|
|
||||||
'action': 'push',
|
|
||||||
'sequence': 23,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
'name': 'WH: 制造后 → 成品存货区',
|
|
||||||
'location_src_id': warehouse.sam_loc_id.id,
|
|
||||||
'location_dest_id': env['stock.location'].search([('name', '=', '成品存货区')], limit=1).id,
|
|
||||||
'route_id': product_route_id.id,
|
|
||||||
'picking_type_id': picking_type_stock.id,
|
|
||||||
'action': 'push',
|
|
||||||
'sequence': 24,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
'name': 'WH: 原料存货区 → 外包位置 (MTO)',
|
|
||||||
'location_src_id': material_location_id.id,
|
|
||||||
'location_dest_id': subcontracting_location_id.id,
|
|
||||||
'route_id': mto_route_id.id,
|
|
||||||
'picking_type_id': picking_type_subcontracting.id,
|
|
||||||
'action': 'pull',
|
|
||||||
'sequence': 24,
|
|
||||||
'warehouse_id': warehouse.id,
|
|
||||||
'procure_method': 'mts_else_mto',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
'name': 'WH: 坯料存货区 → 外包位置 (MTO)',
|
|
||||||
'location_src_id': raw_material_location_id.id,
|
|
||||||
'location_dest_id': subcontracting_location_id.id,
|
|
||||||
'route_id': mto_route_id.id,
|
|
||||||
'picking_type_id': picking_type_subcontracting.id,
|
|
||||||
'action': 'pull',
|
|
||||||
'sequence': 25,
|
|
||||||
'warehouse_id': warehouse.id,
|
|
||||||
'procure_method': 'mts_else_mto',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
'name': 'WH: 坯料存货区 → 外包位置',
|
|
||||||
'location_src_id': raw_material_location_id.id,
|
|
||||||
'location_dest_id': subcontracting_location_id.id,
|
|
||||||
'route_id': resupply_route_id.id,
|
|
||||||
'picking_type_id': picking_type_subcontracting.id,
|
|
||||||
'action': 'pull',
|
|
||||||
'sequence': 26,
|
|
||||||
'warehouse_id': warehouse.id,
|
|
||||||
'procure_method': 'make_to_stock',
|
|
||||||
}
|
|
||||||
]
|
|
||||||
# 遍历每个规则数据,执行创建或更新操作
|
|
||||||
for rule_data in rules_data:
|
|
||||||
_create_or_update_stock_rule(env, rule_data)
|
|
||||||
|
|
||||||
def _create_or_update_stock_rule(env, rule_data):
|
|
||||||
# 尝试查找现有的 stock.rule
|
|
||||||
existing_rule = env['stock.rule'].search([
|
|
||||||
('name', '=', rule_data['name']),
|
|
||||||
('route_id', '=', rule_data.get('route_id'))
|
|
||||||
], limit=1)
|
|
||||||
|
|
||||||
if existing_rule:
|
|
||||||
# 如果存在,更新现有记录
|
|
||||||
existing_rule.write(rule_data)
|
|
||||||
else:
|
|
||||||
# 如果不存在,创建新记录
|
|
||||||
env['stock.rule'].create(rule_data)
|
|
||||||
|
|
||||||
def _data_uninstall(cr, registry):
|
|
||||||
env = api.Environment(cr, SUPERUSER_ID, {})
|
|
||||||
warehouse = env['stock.warehouse'].search([('company_id', '=', env.company.id)], limit=1)
|
|
||||||
product_route_id = warehouse.pbm_route_id
|
|
||||||
resupply_route_id = warehouse.subcontracting_route_id
|
|
||||||
mto_route_id = env.ref('stock.route_warehouse0_mto')
|
|
||||||
# Fail unlink means that the route is used somewhere (e.g. route_id on stock.rule). In this case
|
|
||||||
# we don't try to do anything.
|
|
||||||
try:
|
|
||||||
with env.cr.savepoint():
|
|
||||||
env['stock.rule'].search([('name', 'in', ('WH: 原料存货区 → 制造前', 'WH: 坯料存货区 → 制造前', 'WH: 制造后 → 坯料存货区', 'WH: 制造后 → 成品存货区')), ('route_id', '=', product_route_id.id)]).unlink()
|
|
||||||
env['stock.rule'].search([('name', 'in', ('WH: 原料存货区 → 外包位置 (MTO)', 'WH: 坯料存货区 → 外包位置 (MTO)')), ('route_id', '=', mto_route_id.id)]).unlink()
|
|
||||||
env['stock.rule'].search([('name', '=', 'WH: 坯料存货区 → 外包位置'), ('route_id', '=', resupply_route_id.id)]).unlink()
|
|
||||||
except:
|
|
||||||
pass
|
|
||||||
@@ -1,26 +0,0 @@
|
|||||||
# -*- coding: utf-8 -*-
|
|
||||||
{
|
|
||||||
'name': '机企猫 多供货方式',
|
|
||||||
'version': '16.0.1.0.0',
|
|
||||||
'summary': """ 报价单提供(自动化产线加工/人工线下加工/外购/委外加工)多种供货方式选择 """,
|
|
||||||
'author': 'fox',
|
|
||||||
'website': '',
|
|
||||||
'category': '',
|
|
||||||
'depends': ['sf_dlm', 'sale_stock', 'sf_sale', 'sale'],
|
|
||||||
"data": [
|
|
||||||
'security/ir.model.access.csv',
|
|
||||||
'data/stock_routes.xml',
|
|
||||||
'data/product_data.xml',
|
|
||||||
# 'views/product_product_views.xml',
|
|
||||||
],'assets': {
|
|
||||||
# 'web.assets_backend': [
|
|
||||||
# 'jikimo_sale_multiple_supply_methods/static/src/**/*'
|
|
||||||
# ],
|
|
||||||
},
|
|
||||||
'post_init_hook': '_data_install',
|
|
||||||
'uninstall_hook': '_data_uninstall',
|
|
||||||
'application': True,
|
|
||||||
'installable': True,
|
|
||||||
'auto_install': False,
|
|
||||||
'license': 'LGPL-3',
|
|
||||||
}
|
|
||||||
@@ -1 +0,0 @@
|
|||||||
# -*- coding: utf-8 -*-
|
|
||||||
@@ -1,98 +0,0 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8"?>
|
|
||||||
<odoo>
|
|
||||||
<data noupdate="1">
|
|
||||||
<record id="product_template_manual_processing" model="product.template">
|
|
||||||
<field name="name">人工线下加工模板</field>
|
|
||||||
<field name="active" eval="False"/>
|
|
||||||
<field name="categ_id" ref="sf_dlm.product_category_finished_sf"/>
|
|
||||||
<field name="route_ids"
|
|
||||||
eval="[ref('stock.route_warehouse0_mto'), ref('mrp.route_warehouse0_manufacture')]"/>
|
|
||||||
<field name="invoice_policy">delivery</field>
|
|
||||||
<field name="detailed_type">product</field>
|
|
||||||
<field name="purchase_ok">false</field>
|
|
||||||
<field name="uom_id" ref="uom.product_uom_unit"/>
|
|
||||||
<field name="uom_po_id" ref="uom.product_uom_unit"/>
|
|
||||||
<field name="company_id" ref="base.main_company"/>
|
|
||||||
<field name="single_manufacturing">true</field>
|
|
||||||
<field name="tracking">serial</field>
|
|
||||||
<!-- <field name="categ_type">成品</field> -->
|
|
||||||
<field name="is_manual_processing">true</field>
|
|
||||||
</record>
|
|
||||||
|
|
||||||
<record id="product_template_purchase" model="product.template">
|
|
||||||
<field name="name">成品外购模板</field>
|
|
||||||
<field name="active" eval="False"/>
|
|
||||||
<field name="categ_id" ref="sf_dlm.product_category_finished_sf"/>
|
|
||||||
<field name="route_ids"
|
|
||||||
eval="[ref('stock.route_warehouse0_mto'), ref('purchase_stock.route_warehouse0_buy')]"/>
|
|
||||||
<field name="tracking">serial</field>
|
|
||||||
<field name="detailed_type">product</field>
|
|
||||||
<field name="uom_id" ref="uom.product_uom_unit"/>
|
|
||||||
<field name="uom_po_id" ref="uom.product_uom_unit"/>
|
|
||||||
<field name="company_id" ref="base.main_company"/>
|
|
||||||
<!-- <field name="categ_type">成品</field> -->
|
|
||||||
</record>
|
|
||||||
|
|
||||||
<record id="product_template_outsourcing" model="product.template">
|
|
||||||
<field name="name">成品委外加工模板</field>
|
|
||||||
<field name="active" eval="False"/>
|
|
||||||
<field name="categ_id" ref="sf_dlm.product_category_finished_sf"/>
|
|
||||||
<field name="route_ids"
|
|
||||||
eval="[ref('stock.route_warehouse0_mto'), ref('purchase_stock.route_warehouse0_buy'), ref('mrp_subcontracting.route_resupply_subcontractor_mto')]"/>
|
|
||||||
<field name="tracking">serial</field>
|
|
||||||
<field name="detailed_type">product</field>
|
|
||||||
<field name="uom_id" ref="uom.product_uom_unit"/>
|
|
||||||
<field name="uom_po_id" ref="uom.product_uom_unit"/>
|
|
||||||
<field name="company_id" ref="base.main_company"/>
|
|
||||||
<!-- <field name="categ_type">成品</field> -->
|
|
||||||
</record>
|
|
||||||
|
|
||||||
<record id="product_template_default" model="product.template">
|
|
||||||
<field name="name">成品初始化模板</field>
|
|
||||||
<field name="active" eval="False"/>
|
|
||||||
<field name="categ_id" ref="sf_dlm.product_category_finished_sf"/>
|
|
||||||
<field name="route_ids" eval="[]"/>
|
|
||||||
<field name="tracking">serial</field>
|
|
||||||
<field name="detailed_type">product</field>
|
|
||||||
<field name="uom_id" ref="uom.product_uom_unit"/>
|
|
||||||
<field name="uom_po_id" ref="uom.product_uom_unit"/>
|
|
||||||
<field name="company_id" ref="base.main_company"/>
|
|
||||||
<!-- <field name="categ_type">成品</field> -->
|
|
||||||
</record>
|
|
||||||
|
|
||||||
<!-- 供应商信息(业务平台),由于数据是python创建,只能指定ID -->
|
|
||||||
<record id="product_supplierinfo_bfm" model="product.supplierinfo">
|
|
||||||
<field name="partner_id" eval="91"/>
|
|
||||||
</record>
|
|
||||||
|
|
||||||
<record id="product_template_embryo_customer_provided" model="product.template">
|
|
||||||
<field name="name">坯料客供料模板</field>
|
|
||||||
<field name="active" eval="False"/>
|
|
||||||
<field name="categ_id" ref="sf_dlm.product_category_embryo_sf"/>
|
|
||||||
<field name="route_ids" eval="[
|
|
||||||
ref('stock.route_warehouse0_mto'),
|
|
||||||
ref('mrp_subcontracting.route_resupply_subcontractor_mto'),
|
|
||||||
ref('jikimo_sale_multiple_supply_methods.route_material_processing')]"/>
|
|
||||||
<field name="sale_ok">false</field>
|
|
||||||
<field name="tracking">serial</field>
|
|
||||||
<field name="detailed_type">product</field>
|
|
||||||
<field name="uom_id" ref="uom.product_uom_unit"/>
|
|
||||||
<field name="uom_po_id" ref="uom.product_uom_unit"/>
|
|
||||||
<field name="company_id" ref="base.main_company"/>
|
|
||||||
<!-- <field name="categ_type">坯料</field> -->
|
|
||||||
<field name="seller_ids" eval="[ref('jikimo_sale_multiple_supply_methods.product_supplierinfo_bfm')]"/>
|
|
||||||
</record>
|
|
||||||
|
|
||||||
<record id="sf_dlm.product_embryo_sf_self_machining" model="product.product">
|
|
||||||
<field name="is_manual_processing">true</field>
|
|
||||||
</record>
|
|
||||||
|
|
||||||
<record id="sf_dlm.product_embryo_sf_self_machining" model="product.product">
|
|
||||||
<field name="route_ids" eval="[(4, ref('mrp_subcontracting.route_resupply_subcontractor_mto'))]"/>
|
|
||||||
</record>
|
|
||||||
|
|
||||||
<record id="sf_dlm.product_embryo_sf_purchase" model="product.product">
|
|
||||||
<field name="route_ids" eval="[(4, ref('mrp_subcontracting.route_resupply_subcontractor_mto'))]"/>
|
|
||||||
</record>
|
|
||||||
</data>
|
|
||||||
</odoo>
|
|
||||||
@@ -1,32 +0,0 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8"?>
|
|
||||||
<odoo>
|
|
||||||
<data noupdate="1">
|
|
||||||
<record id="route_material_processing" model="stock.route">
|
|
||||||
<field name="name">带料加工</field>
|
|
||||||
<field name="product_selectable">true</field>
|
|
||||||
<field name="warehouse_selectable">true</field>
|
|
||||||
<field name="warehouse_ids" eval="[ref('stock.warehouse0')]"/>
|
|
||||||
<field name="sequence">16</field>
|
|
||||||
</record>
|
|
||||||
|
|
||||||
<record id="material_picking_in" model="stock.picking.type">
|
|
||||||
<field name="name">客供料入库</field>
|
|
||||||
<field name="code">incoming</field>
|
|
||||||
<field name="active">true</field>
|
|
||||||
<field name="company_id" ref="base.main_company"/>
|
|
||||||
<field name="sequence_code">DL</field>
|
|
||||||
<field name="warehouse_id" ref="stock.warehouse0"/>
|
|
||||||
<field name="default_location_src_id" ref="stock.stock_location_customers"/>
|
|
||||||
<field name="default_location_dest_id" eval="25"/>
|
|
||||||
</record>
|
|
||||||
|
|
||||||
<record id="rule_material_receiving" model="stock.rule">
|
|
||||||
<field name="name">带料收货</field>
|
|
||||||
<field name="route_id" ref="route_material_processing"/>
|
|
||||||
<field name="location_dest_id" ref="stock.stock_location_company"/>
|
|
||||||
<field name="location_src_id" ref="stock.stock_location_customers"/>
|
|
||||||
<field name="picking_type_id" ref="material_picking_in"/>
|
|
||||||
<field name="action">pull</field>
|
|
||||||
</record>
|
|
||||||
</data>
|
|
||||||
</odoo>
|
|
||||||
@@ -1,3 +0,0 @@
|
|||||||
# -*- coding: utf-8 -*-
|
|
||||||
from . import product_template
|
|
||||||
from . import mrp_bom
|
|
||||||
@@ -1,14 +0,0 @@
|
|||||||
from odoo import models, fields
|
|
||||||
|
|
||||||
class MrpBom(models.Model):
|
|
||||||
_inherit = 'mrp.bom'
|
|
||||||
|
|
||||||
# 业务平台分配工厂后在智能工厂先创建销售订单再创建该产品后再次进行创建bom
|
|
||||||
def bom_create(self, product, bom_type, product_type):
|
|
||||||
bom_id = super(MrpBom, self).bom_create(product, bom_type, product_type)
|
|
||||||
|
|
||||||
# 成品的供应商从模板中获取
|
|
||||||
if product_type == 'product':
|
|
||||||
if product.product_tmpl_id.seller_ids:
|
|
||||||
bom_id.subcontractor_id = product.product_tmpl_id.seller_ids[-1].partner_id.id
|
|
||||||
return bom_id
|
|
||||||
@@ -1,26 +0,0 @@
|
|||||||
from odoo import models, fields, api
|
|
||||||
|
|
||||||
class ProductTemplate(models.Model):
|
|
||||||
_inherit = 'product.template'
|
|
||||||
|
|
||||||
is_manual_processing = fields.Boolean(string='人工线下加工')
|
|
||||||
is_customer_provided = fields.Boolean(string='客供料')
|
|
||||||
|
|
||||||
def copy_template(self, product_template_id):
|
|
||||||
if not isinstance(product_template_id, ProductTemplate):
|
|
||||||
raise ValueError('%s必须是ProductTemplate类型' % product_template_id)
|
|
||||||
|
|
||||||
self.route_ids = product_template_id.route_ids
|
|
||||||
self.categ_id = product_template_id.categ_id
|
|
||||||
self.invoice_policy = product_template_id.invoice_policy
|
|
||||||
self.detailed_type = product_template_id.detailed_type
|
|
||||||
self.purchase_ok = product_template_id.purchase_ok
|
|
||||||
self.uom_id = product_template_id.uom_id
|
|
||||||
self.uom_po_id = product_template_id.uom_po_id
|
|
||||||
self.company_id = product_template_id.company_id
|
|
||||||
self.single_manufacturing = product_template_id.single_manufacturing
|
|
||||||
self.tracking = product_template_id.tracking
|
|
||||||
self.is_bfm = product_template_id.is_bfm
|
|
||||||
self.is_manual_processing = product_template_id.is_manual_processing
|
|
||||||
# 复制 seller_ids
|
|
||||||
self.seller_ids = [(0, 0, {'partner_id': seller.partner_id.id, 'delay': 1.0, 'price': seller.price}) for seller in product_template_id.seller_ids]
|
|
||||||
@@ -1,8 +0,0 @@
|
|||||||
id,name,model_id:id,group_id:id,perm_read,perm_write,perm_create,perm_unlink
|
|
||||||
access_sale_order_group_production_engineer,sale.order_group_production_engineer,sale.model_sale_order,sf_base.group_production_engineer,1,1,0,0
|
|
||||||
access_sale_order_line_group_production_engineer,sale_order_line_group_production_engineer,sale.model_sale_order_line,sf_base.group_production_engineer,1,1,0,0
|
|
||||||
access_product_product_group_production_engineer,product_product_group_production_engineer,product.model_product_product,sf_base.group_production_engineer,1,0,0,0
|
|
||||||
access_product_template_group_production_engineer,product_template_group_production_engineer,product.model_product_template,sf_base.group_production_engineer,1,0,0,0
|
|
||||||
access_stock_picking_group_production_engineer,stock_picking_group_production_engineer,stock.model_stock_picking,sf_base.group_production_engineer,1,0,0,0
|
|
||||||
access_stock_move_group_production_engineer,stock_move_group_production_engineer,stock.model_stock_move,sf_base.group_production_engineer,1,0,0,0
|
|
||||||
access_mrp_bom_group_production_engineer,mrp_bom_group_production_engineer,mrp.model_mrp_bom,sf_base.group_production_engineer,1,0,0,0
|
|
||||||
|
@@ -1,16 +0,0 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8" ?>
|
|
||||||
<odoo>
|
|
||||||
<!-- 由于该模块不能依赖sf_dlm_management, 该功能只能在sf_dlm_management中实现,并且依赖该模块-->
|
|
||||||
<record id="view_product_product_form_inherit_sf" model="ir.ui.view">
|
|
||||||
<field name="name">view.product.template.form.inherit.sf</field>
|
|
||||||
<field name="model">product.template</field>
|
|
||||||
<field name="inherit_id" ref="sf_dlm_management.view_sale_product_template_form_inherit_sf"/>
|
|
||||||
<field name="arch" type="xml">
|
|
||||||
<xpath expr="//field[@name='manual_quotation']" position="after">
|
|
||||||
<field name="is_customer_provided" attrs="{'invisible': [('categ_type', 'not in', ['成品', '坯料'])], 'readonly': True}" />
|
|
||||||
</xpath>
|
|
||||||
</field>
|
|
||||||
</record>
|
|
||||||
</odoo>
|
|
||||||
|
|
||||||
|
|
||||||
@@ -1,3 +0,0 @@
|
|||||||
from . import models
|
|
||||||
from . import controllers
|
|
||||||
from . import wizards
|
|
||||||
@@ -1,32 +0,0 @@
|
|||||||
{
|
|
||||||
'name': '机企猫 测试助手',
|
|
||||||
'version': '16.0.1.0.0',
|
|
||||||
'category': 'Technical',
|
|
||||||
'summary': '测试数据初始化工具',
|
|
||||||
'description': """
|
|
||||||
用于初始化测试环境数据的工具模块
|
|
||||||
""",
|
|
||||||
'author': 'Jikimo',
|
|
||||||
'website': 'www.jikimo.com',
|
|
||||||
'depends': [
|
|
||||||
'base',
|
|
||||||
'sale_management',
|
|
||||||
'purchase',
|
|
||||||
'mrp',
|
|
||||||
'stock',
|
|
||||||
'account'
|
|
||||||
],
|
|
||||||
'data': [
|
|
||||||
'security/ir.model.access.csv',
|
|
||||||
'wizards/jikimo_data_clean_wizard.xml',
|
|
||||||
],
|
|
||||||
'assets': {
|
|
||||||
'web.assets_backend': [
|
|
||||||
'jikimo_test_assistant/static/src/js/data_clean_confirm.js',
|
|
||||||
],
|
|
||||||
},
|
|
||||||
'installable': True,
|
|
||||||
'application': False,
|
|
||||||
'auto_install': False,
|
|
||||||
'license': 'LGPL-3',
|
|
||||||
}
|
|
||||||
@@ -1,2 +0,0 @@
|
|||||||
# -*- coding: utf-8 -*-
|
|
||||||
from . import main
|
|
||||||
@@ -1,86 +0,0 @@
|
|||||||
from odoo import http
|
|
||||||
import logging
|
|
||||||
import os
|
|
||||||
import json
|
|
||||||
import sys
|
|
||||||
|
|
||||||
_logger = logging.getLogger(__name__)
|
|
||||||
|
|
||||||
class Main(http.Controller):
|
|
||||||
@http.route('/api/pdf2image', type='http', auth='public', methods=['POST'], csrf=False)
|
|
||||||
def convert_pdf_to_image(self, **kwargs):
|
|
||||||
"""将PDF文件转换为图片文件
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
dict: 包含转换后图片url的字典
|
|
||||||
"""
|
|
||||||
res = {}
|
|
||||||
try:
|
|
||||||
# 检查poppler是否可用
|
|
||||||
# if sys.platform.startswith('win'):
|
|
||||||
# if not os.environ.get('POPPLER_PATH'):
|
|
||||||
# return {
|
|
||||||
# 'code': 400,
|
|
||||||
# 'msg': '请先配置POPPLER_PATH环境变量'
|
|
||||||
# }
|
|
||||||
# else:
|
|
||||||
# import shutil
|
|
||||||
# if not shutil.which('pdftoppm'):
|
|
||||||
# return {
|
|
||||||
# 'code': 400,
|
|
||||||
# 'msg': '请先安装poppler-utils'
|
|
||||||
# }
|
|
||||||
|
|
||||||
# 获取上传的PDF文件
|
|
||||||
pdf_file = kwargs.get('file')
|
|
||||||
if not pdf_file:
|
|
||||||
res = {'code': 400, 'msg': '未找到上传的PDF文件'}
|
|
||||||
|
|
||||||
# 检查文件类型
|
|
||||||
if not pdf_file.filename.lower().endswith('.pdf'):
|
|
||||||
res = {'code': 400, 'msg': '请上传PDF格式的文件'}
|
|
||||||
|
|
||||||
# 读取PDF文件内容
|
|
||||||
pdf_content = pdf_file.read()
|
|
||||||
|
|
||||||
# 使用pdf2image转换
|
|
||||||
from pdf2image import convert_from_bytes
|
|
||||||
import tempfile
|
|
||||||
|
|
||||||
# 转换PDF
|
|
||||||
with tempfile.TemporaryDirectory() as path:
|
|
||||||
images = convert_from_bytes(pdf_content)
|
|
||||||
image_urls = []
|
|
||||||
|
|
||||||
# 保存每一页为图片
|
|
||||||
for i, image in enumerate(images):
|
|
||||||
image_path = os.path.join(path, f'page_{i+1}.jpg')
|
|
||||||
image.save(image_path, 'JPEG')
|
|
||||||
|
|
||||||
# 将图片保存到ir.attachment
|
|
||||||
with open(image_path, 'rb') as img_file:
|
|
||||||
attachment = http.request.env['ir.attachment'].sudo().create({
|
|
||||||
'name': f'page_{i+1}.jpg',
|
|
||||||
'datas': img_file.read(),
|
|
||||||
'type': 'binary',
|
|
||||||
'access_token': kwargs.get('access_token') or '123'
|
|
||||||
})
|
|
||||||
image_urls.append({
|
|
||||||
'page': i+1,
|
|
||||||
'url': f'/web/content/{attachment.id}'
|
|
||||||
})
|
|
||||||
|
|
||||||
res = {
|
|
||||||
'code': 200,
|
|
||||||
'msg': '转换成功',
|
|
||||||
'data': image_urls
|
|
||||||
}
|
|
||||||
|
|
||||||
except Exception as e:
|
|
||||||
_logger.error('PDF转换失败: %s', str(e))
|
|
||||||
res = {
|
|
||||||
'code': 500,
|
|
||||||
'msg': f'转换失败: {str(e)}'
|
|
||||||
}
|
|
||||||
return json.JSONEncoder().encode(res)
|
|
||||||
|
|
||||||
@@ -1 +0,0 @@
|
|||||||
|
|
||||||
@@ -1,2 +0,0 @@
|
|||||||
id,name,model_id:id,group_id:id,perm_read,perm_write,perm_create,perm_unlink
|
|
||||||
access_jikimo_data_clean_wizard,jikimo_test_assistant.jikimo_data_clean_wizard,model_jikimo_data_clean_wizard,base.group_system,1,1,1,1
|
|
||||||
|
@@ -1,50 +0,0 @@
|
|||||||
odoo.define('jikimo_test_assistant.action_clean_data_confirm', function (require) {
|
|
||||||
const core = require('web.core');
|
|
||||||
const ajax = require('web.ajax');
|
|
||||||
const Dialog = require('web.Dialog');
|
|
||||||
var rpc = require('web.rpc');
|
|
||||||
var _t = core._t;
|
|
||||||
|
|
||||||
async function action_clean_data_confirm(parent, {params}) {
|
|
||||||
let message = "确认清理数据?<br/>"
|
|
||||||
message += "日期:"+ params.date + "以前<br/>"
|
|
||||||
message += "模型:" + params.model_names.join(',')
|
|
||||||
const dialog = new Dialog(parent, {
|
|
||||||
title: "确认",
|
|
||||||
$content: $('<div>').append(message),
|
|
||||||
buttons: [
|
|
||||||
{ text: "确认", classes: 'btn-primary jikimo_button_confirm', close: true, click: () => actionCleanDataConfirm(parent, params) },
|
|
||||||
{ text: "取消", close: true },
|
|
||||||
],
|
|
||||||
});
|
|
||||||
dialog.open();
|
|
||||||
|
|
||||||
|
|
||||||
async function actionCleanDataConfirm(parent, params) {
|
|
||||||
rpc.query({
|
|
||||||
model: 'jikimo.data.clean.wizard',
|
|
||||||
method: 'action_clean_data',
|
|
||||||
args: [params.active_id],
|
|
||||||
kwargs: {
|
|
||||||
context: params.context,
|
|
||||||
}
|
|
||||||
}).then(res => {
|
|
||||||
parent.services.action.doAction({
|
|
||||||
'type': 'ir.actions.client',
|
|
||||||
'tag': 'display_notification',
|
|
||||||
'target': 'new',
|
|
||||||
'params': {
|
|
||||||
'message': '数据清理成功!',
|
|
||||||
'type': 'success',
|
|
||||||
'sticky': false,
|
|
||||||
'next': {'type': 'ir.actions.act_window_close'},
|
|
||||||
}
|
|
||||||
});
|
|
||||||
})
|
|
||||||
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
core.action_registry.add('action_clean_data_confirm', action_clean_data_confirm);
|
|
||||||
return action_clean_data_confirm;
|
|
||||||
});
|
|
||||||
@@ -1,2 +0,0 @@
|
|||||||
# -*- coding: utf-8 -*-
|
|
||||||
from . import jikimo_data_clean_wizard
|
|
||||||
@@ -1,99 +0,0 @@
|
|||||||
from odoo import models, fields, api
|
|
||||||
from datetime import datetime
|
|
||||||
|
|
||||||
import logging
|
|
||||||
|
|
||||||
_logger = logging.getLogger(__name__)
|
|
||||||
|
|
||||||
class JikimoDataCleanWizard(models.TransientModel):
|
|
||||||
_name = 'jikimo.data.clean.wizard'
|
|
||||||
_description = '业务数据清理'
|
|
||||||
|
|
||||||
date = fields.Date(string='截止日期', required=True, default=fields.Date.context_today)
|
|
||||||
model_ids = fields.Many2many('ir.model', string='业务模型', domain=[
|
|
||||||
('model', 'in', [
|
|
||||||
'sale.order', # 销售订单
|
|
||||||
'purchase.order', # 采购订单
|
|
||||||
'mrp.production', # 生产订单
|
|
||||||
'stock.picking', # 库存调拨
|
|
||||||
'account.move', # 会计凭证
|
|
||||||
])
|
|
||||||
])
|
|
||||||
|
|
||||||
def action_clean_data(self):
|
|
||||||
self.ensure_one()
|
|
||||||
model_list = self.model_ids.mapped('model')
|
|
||||||
|
|
||||||
# 销售订单清理(排除已交付,已锁定,已取消)
|
|
||||||
if 'sale.order' in model_list:
|
|
||||||
self.model_cancel('sale.order', except_states=['delivered', 'done', 'cancel'])
|
|
||||||
|
|
||||||
# 采购订单清理(排除采购订单,已锁定,已取消)
|
|
||||||
if 'purchase.order' in model_list:
|
|
||||||
self.model_cancel('purchase.order', except_states=['purchase', 'done', 'cancel'])
|
|
||||||
|
|
||||||
# 生产订单清理(排除返工,报废,完成,已取消)
|
|
||||||
if 'mrp.production' in model_list:
|
|
||||||
self.model_cancel('mrp.production', except_states=['rework', 'scrap', 'done', 'cancel'])
|
|
||||||
|
|
||||||
# 工单清理 (排除返工,完成,已取消)
|
|
||||||
if 'mrp.workorder' in model_list:
|
|
||||||
self.model_cancel('mrp.production', except_states=['rework', 'done', 'cancel'])
|
|
||||||
|
|
||||||
# 排程单清理 (排除已完成,已取消)
|
|
||||||
if 'mrp.workorder' in model_list:
|
|
||||||
self.model_cancel('mrp.production', except_states=['finished', 'cancel'])
|
|
||||||
|
|
||||||
# 工单库存移动 (排除完成,已取消)
|
|
||||||
if 'stock.move' in model_list:
|
|
||||||
self.model_cancel('stock.move')
|
|
||||||
|
|
||||||
# 库存调拨清理 (排除完成,已取消)
|
|
||||||
if 'stock.picking' in model_list:
|
|
||||||
self.model_cancel('stock.picking')
|
|
||||||
|
|
||||||
# 会计凭证清理 (排除已过账,已取消)
|
|
||||||
if 'account.move' in model_list:
|
|
||||||
self.model_cancel('account.move', except_states=['posted', 'cancel'])
|
|
||||||
|
|
||||||
return True
|
|
||||||
|
|
||||||
def model_cancel(self, model_name, state_field='state', to_state='cancel',except_states=('done', 'cancel')):
|
|
||||||
table = self.env[model_name]._table
|
|
||||||
if isinstance(except_states, list):
|
|
||||||
except_states = tuple(except_states)
|
|
||||||
sql = """
|
|
||||||
UPDATE
|
|
||||||
%s SET %s = '%s'
|
|
||||||
WHERE
|
|
||||||
create_date < '%s'
|
|
||||||
AND state NOT IN %s
|
|
||||||
""" % (table, state_field, to_state, self.date.strftime('%Y-%m-%d'), except_states)
|
|
||||||
self.env.cr.execute(sql)
|
|
||||||
self.env.cr.commit()
|
|
||||||
|
|
||||||
@api.model
|
|
||||||
def get_confirm_message(self):
|
|
||||||
date_str = self.date.strftime('%Y-%m-%d') if self.date else ''
|
|
||||||
model_names = ', '.join([model.name for model in self.model_ids])
|
|
||||||
return {
|
|
||||||
'date': date_str,
|
|
||||||
'model_names': model_names
|
|
||||||
}
|
|
||||||
|
|
||||||
def action_clean_data_confirm(self):
|
|
||||||
model_names = self.model_ids.mapped('display_name')
|
|
||||||
return {
|
|
||||||
'type': 'ir.actions.client',
|
|
||||||
'tag': 'action_clean_data_confirm',
|
|
||||||
'params': {
|
|
||||||
'model_names': model_names,
|
|
||||||
'date': self.date,
|
|
||||||
'active_id': self.id,
|
|
||||||
'context': self.env.context
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
@@ -1,47 +0,0 @@
|
|||||||
<?xml version="1.0" encoding="utf-8"?>
|
|
||||||
<odoo>
|
|
||||||
<!-- Form View -->
|
|
||||||
<record id="view_jikimo_data_clean_form" model="ir.ui.view">
|
|
||||||
<field name="name">jikimo.data.clean.wizard.form</field>
|
|
||||||
<field name="model">jikimo.data.clean.wizard</field>
|
|
||||||
<field name="arch" type="xml">
|
|
||||||
<form string="业务数据清理">
|
|
||||||
<sheet>
|
|
||||||
<group>
|
|
||||||
<field name="date"/>
|
|
||||||
<field name="model_ids" widget="many2many_tags"/>
|
|
||||||
</group>
|
|
||||||
</sheet>
|
|
||||||
<footer>
|
|
||||||
<button name="action_clean_data_confirm"
|
|
||||||
string="确认清理"
|
|
||||||
type="object"
|
|
||||||
class="btn-primary"/>
|
|
||||||
<button special="cancel"
|
|
||||||
string="取消"
|
|
||||||
class="btn-secondary"/>
|
|
||||||
</footer>
|
|
||||||
</form>
|
|
||||||
</field>
|
|
||||||
</record>
|
|
||||||
|
|
||||||
<!-- Action -->
|
|
||||||
<record id="action_jikimo_data_clean" model="ir.actions.act_window">
|
|
||||||
<field name="name">业务数据清理</field>
|
|
||||||
<field name="res_model">jikimo.data.clean.wizard</field>
|
|
||||||
<field name="view_mode">form</field>
|
|
||||||
<field name="target">new</field>
|
|
||||||
</record>
|
|
||||||
|
|
||||||
<!-- Menu -->
|
|
||||||
<menuitem id="menu_test_root"
|
|
||||||
name="测试"
|
|
||||||
parent="base.menu_custom"
|
|
||||||
sequence="100"/>
|
|
||||||
|
|
||||||
<menuitem id="menu_jikimo_data_clean"
|
|
||||||
name="业务数据清理"
|
|
||||||
parent="menu_test_root"
|
|
||||||
action="action_jikimo_data_clean"
|
|
||||||
sequence="10"/>
|
|
||||||
</odoo>
|
|
||||||
@@ -1,3 +0,0 @@
|
|||||||
# -*- coding: utf-8 -*-
|
|
||||||
from . import controllers
|
|
||||||
from . import models
|
|
||||||
@@ -1,17 +0,0 @@
|
|||||||
# -*- coding: utf-8 -*-
|
|
||||||
{
|
|
||||||
'name': '机企猫 报工系统API',
|
|
||||||
'version': '1.0.0',
|
|
||||||
'summary': """ 机企猫 报工系统API """,
|
|
||||||
'author': '机企猫',
|
|
||||||
'website': 'https://xt.sf.jikimo.com',
|
|
||||||
'category': 'sf',
|
|
||||||
'depends': ['base', 'sf_maintenance', 'jikimo_mini_program'],
|
|
||||||
'data': [
|
|
||||||
],
|
|
||||||
|
|
||||||
'application': True,
|
|
||||||
'installable': True,
|
|
||||||
'auto_install': False,
|
|
||||||
'license': 'LGPL-3',
|
|
||||||
}
|
|
||||||
@@ -1,2 +0,0 @@
|
|||||||
# -*- coding: utf-8 -*-
|
|
||||||
from . import main
|
|
||||||
@@ -1,69 +0,0 @@
|
|||||||
import json
|
|
||||||
from odoo import http
|
|
||||||
from odoo.http import request
|
|
||||||
from odoo.addons.sf_machine_connect.models.ftp_operate import transfer_files
|
|
||||||
from odoo.addons.sf_base.decorators.api_log import api_log
|
|
||||||
|
|
||||||
class MainController(http.Controller):
|
|
||||||
|
|
||||||
@http.route('/api/manual_download_program', type='json', methods=['POST'], auth='wechat_token', cors='*')
|
|
||||||
@api_log('人工线下加工编程文件传输', requester='报工系统')
|
|
||||||
def manual_download_program(self):
|
|
||||||
"""
|
|
||||||
人工线下加工传输编程文件
|
|
||||||
"""
|
|
||||||
data = json.loads(request.httprequest.data)
|
|
||||||
maintenance_equipment_id = data.get('maintenance_equipment_id')
|
|
||||||
model_id = data.get('model_id')
|
|
||||||
if not maintenance_equipment_id or not model_id:
|
|
||||||
return {'code': 400, 'message': '参数错误'}
|
|
||||||
try:
|
|
||||||
model_id = int(model_id)
|
|
||||||
except Exception as e:
|
|
||||||
return {'code': 400, 'message': '参数类型错误'}
|
|
||||||
maintenance_equipment = request.env['maintenance.equipment'].sudo().search(
|
|
||||||
[('MTcode', '=', maintenance_equipment_id), ('category_id.equipment_type', '=', '机床')],
|
|
||||||
limit=1
|
|
||||||
)
|
|
||||||
if not maintenance_equipment:
|
|
||||||
return {'code': 400, 'message': '机台不存在,请扫描正确的机台二维码'}
|
|
||||||
product = request.env['product.template'].sudo().search([('model_id', '=', model_id)], limit=1)
|
|
||||||
if not product:
|
|
||||||
return {'code': 400, 'message': '请扫描正确的图纸'}
|
|
||||||
# 获取刀具组
|
|
||||||
tool_groups_id = request.env['sf.tool.groups'].sudo().search([('equipment_ids', 'in', maintenance_equipment.id)], limit=1)
|
|
||||||
if not tool_groups_id:
|
|
||||||
return {'code': 400, 'message': '刀具组不存在'}
|
|
||||||
|
|
||||||
ftp_resconfig = request.env['res.config.settings'].sudo().get_values()
|
|
||||||
if not ftp_resconfig['ftp_host'] or not ftp_resconfig['ftp_port'] or not ftp_resconfig['ftp_user'] or not ftp_resconfig['ftp_password']:
|
|
||||||
return {'code': 400, 'message': '编程文件FTP配置错误'}
|
|
||||||
source_ftp_info = {
|
|
||||||
'host': ftp_resconfig['ftp_host'],
|
|
||||||
'port': int(ftp_resconfig['ftp_port']),
|
|
||||||
'username': ftp_resconfig['ftp_user'],
|
|
||||||
'password': ftp_resconfig['ftp_password']
|
|
||||||
}
|
|
||||||
if not maintenance_equipment.ftp_host or not maintenance_equipment.ftp_port or not maintenance_equipment.ftp_username or not maintenance_equipment.ftp_password:
|
|
||||||
return {'code': 400, 'message': '机台FTP配置错误'}
|
|
||||||
target_ftp_info = {
|
|
||||||
'host': maintenance_equipment.ftp_host,
|
|
||||||
'port': int(maintenance_equipment.ftp_port),
|
|
||||||
'username': maintenance_equipment.ftp_username,
|
|
||||||
'password': maintenance_equipment.ftp_password
|
|
||||||
}
|
|
||||||
# 传输nc文件
|
|
||||||
try:
|
|
||||||
result = transfer_files(
|
|
||||||
source_ftp_info,
|
|
||||||
target_ftp_info,
|
|
||||||
'/' + str(model_id),
|
|
||||||
'/',
|
|
||||||
match_str=r'^\d*-' + tool_groups_id.name + r'-\w{2}-all\.nc$'
|
|
||||||
)
|
|
||||||
if len(result) > 0:
|
|
||||||
return {'code': 200, 'message': '传输成功', 'file_list': result}
|
|
||||||
else:
|
|
||||||
return {'code': 404, 'message': '未找到编程文件'}
|
|
||||||
except Exception as e:
|
|
||||||
return {'code': 500, 'message': str(e)}
|
|
||||||
@@ -1 +0,0 @@
|
|||||||
# -*- coding: utf-8 -*-
|
|
||||||
@@ -4,8 +4,6 @@ import json
|
|||||||
import logging
|
import logging
|
||||||
from odoo.addons.sf_mrs_connect.controllers.controllers import Sf_Mrs_Connect
|
from odoo.addons.sf_mrs_connect.controllers.controllers import Sf_Mrs_Connect
|
||||||
from odoo.addons.sf_manufacturing.controllers.controllers import Manufacturing_Connect
|
from odoo.addons.sf_manufacturing.controllers.controllers import Manufacturing_Connect
|
||||||
from odoo.addons.sf_base.decorators.api_log import api_log
|
|
||||||
from datetime import datetime
|
|
||||||
|
|
||||||
_logger = logging.getLogger(__name__)
|
_logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
@@ -13,7 +11,6 @@ class WorkorderExceptionConroller(http.Controller):
|
|||||||
|
|
||||||
@http.route('/AutoDeviceApi/BillError', type='json', auth='public', methods=['GET', 'POST'], csrf=False,
|
@http.route('/AutoDeviceApi/BillError', type='json', auth='public', methods=['GET', 'POST'], csrf=False,
|
||||||
cors="*")
|
cors="*")
|
||||||
@api_log('工单对接错误', requester='中控系统')
|
|
||||||
def workder_exception(self, **kw):
|
def workder_exception(self, **kw):
|
||||||
"""
|
"""
|
||||||
记录工单异常
|
记录工单异常
|
||||||
@@ -33,7 +30,6 @@ class WorkorderExceptionConroller(http.Controller):
|
|||||||
workorder = request.env['mrp.workorder'].sudo().search([
|
workorder = request.env['mrp.workorder'].sudo().search([
|
||||||
('rfid_code', '=', ret['RfidCode']),
|
('rfid_code', '=', ret['RfidCode']),
|
||||||
('routing_type', '=', 'CNC加工'),
|
('routing_type', '=', 'CNC加工'),
|
||||||
('state', '!=', 'rework')
|
|
||||||
])
|
])
|
||||||
if not workorder:
|
if not workorder:
|
||||||
res = {'Succeed': False, 'ErrorCode': 401, 'Error': '无效的工单'}
|
res = {'Succeed': False, 'ErrorCode': 401, 'Error': '无效的工单'}
|
||||||
@@ -45,12 +41,9 @@ class WorkorderExceptionConroller(http.Controller):
|
|||||||
'exception_code': ret.get('coding'),
|
'exception_code': ret.get('coding'),
|
||||||
'exception_content': ret.get('Error', '')
|
'exception_content': ret.get('Error', '')
|
||||||
})
|
})
|
||||||
# 申请重新编程
|
|
||||||
workorder.production_id.update_programming_state(trigger_time=datetime.now().strftime('%Y-%m-%d %H:%M:%S'),
|
|
||||||
reprogramming_reason=ret.get('Error', ''))
|
|
||||||
workorder.production_id.write({'programming_state': '编程中', 'work_state': '编程中', 'is_rework': False})
|
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
res = {'Succeed': False, 'ErrorCode': 202, 'Error': e}
|
res = {'Succeed': False, 'ErrorCode': 202, 'Error': str(e)}
|
||||||
_logger.info('workder_exception error:%s' % e)
|
_logger.info('workder_exception error:%s' % e)
|
||||||
return json.JSONEncoder().encode(res)
|
return json.JSONEncoder().encode(res)
|
||||||
|
|
||||||
|
|||||||
@@ -1,5 +1,2 @@
|
|||||||
"id","name","model_id:id","group_id:id","perm_read","perm_write","perm_create","perm_unlink"
|
"id","name","model_id:id","group_id:id","perm_read","perm_write","perm_create","perm_unlink"
|
||||||
"access_jikimo_workorder_exception","access.jikimo.workorder.exception","model_jikimo_workorder_exception","mrp.group_mrp_user",1,1,1,0
|
"access_jikimo_workorder_exception","access.jikimo.workorder.exception","model_jikimo_workorder_exception","mrp.group_mrp_user",1,1,1,0
|
||||||
"access_jikimo_workorder_exception_group_quality","access.jikimo.workorder.exception.group_quality","model_jikimo_workorder_exception","sf_base.group_quality",1,1,1,0
|
|
||||||
"access_jikimo_workorder_exception_group_quality_director","access.jikimo.workorder.exception.group_quality_director","model_jikimo_workorder_exception","sf_base.group_quality_director",1,1,1,0
|
|
||||||
|
|
||||||
|
|||||||
|
@@ -7,8 +7,7 @@
|
|||||||
<field name="inherit_id" ref="mrp.mrp_production_workorder_form_view_inherit"/>
|
<field name="inherit_id" ref="mrp.mrp_production_workorder_form_view_inherit"/>
|
||||||
<field name="arch" type="xml">
|
<field name="arch" type="xml">
|
||||||
<xpath expr="//notebook/page[last()]" position="after">
|
<xpath expr="//notebook/page[last()]" position="after">
|
||||||
<field name="routing_type" invisible="1"/>
|
<page string="异常记录" name="workorder_exception" attrs="{'invisible': [('routing_type', '!=', 'CNC加工')]}">
|
||||||
<page string="异常记录" name="workorder_exception" attrs='{"invisible": ["!", ("individuation_page_list", "ilike", "ER")]}'>
|
|
||||||
<field name="exception_ids" nolabel="1" readonly="1">
|
<field name="exception_ids" nolabel="1" readonly="1">
|
||||||
<tree create="false" delete="false" edit="false">
|
<tree create="false" delete="false" edit="false">
|
||||||
<field name="exception_content" string="反馈的异常/问题信息"/>
|
<field name="exception_content" string="反馈的异常/问题信息"/>
|
||||||
|
|||||||
@@ -51,11 +51,11 @@ class JikimoWorkorderException(models.Model):
|
|||||||
return res
|
return res
|
||||||
|
|
||||||
def _get_message(self, message_queue_ids):
|
def _get_message(self, message_queue_ids):
|
||||||
contents, _ = super(JikimoWorkorderException, self)._get_message(message_queue_ids)
|
contents = super(JikimoWorkorderException, self)._get_message(message_queue_ids)
|
||||||
base_url = self.env['ir.config_parameter'].get_param('web.base.url')
|
url = self.env['ir.config_parameter'].get_param('web.base.url')
|
||||||
action_id = self.env.ref('mrp.mrp_production_action').id
|
action_id = self.env.ref('mrp.mrp_production_action').id
|
||||||
for index, content in enumerate(contents):
|
for index, content in enumerate(contents):
|
||||||
exception_id = self.env['jikimo.workorder.exception'].browse(message_queue_ids[index].res_id)
|
exception_id = self.env['jikimo.workorder.exception'].browse(message_queue_ids[index].res_id)
|
||||||
url = base_url + '/web#id=%s&view_type=form&action=%s' % (exception_id.workorder_id.production_id.id, action_id)
|
url = url + '/web#id=%s&view_type=form&action=%s' % (exception_id.workorder_id.production_id.id, action_id)
|
||||||
contents[index] = content.replace('{{url}}', url)
|
contents[index] = content.replace('{{url}}', url)
|
||||||
return contents, message_queue_ids
|
return contents
|
||||||
|
|||||||
@@ -1273,18 +1273,3 @@ msgstr ""
|
|||||||
#: model:product.template,description_sale:mrp_workorder.product_template_stool_top
|
#: model:product.template,description_sale:mrp_workorder.product_template_stool_top
|
||||||
msgid "wooden stool top"
|
msgid "wooden stool top"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
#. module: mrp_workorder
|
|
||||||
#: model:quality.point.test_type,name:mrp_workorder.test_type_register_consumed_materials
|
|
||||||
msgid "Register Consumed Materials"
|
|
||||||
msgstr "登记消耗材料"
|
|
||||||
|
|
||||||
#. module: mrp_workorder
|
|
||||||
#: model:quality.point.test_type,name:mrp_workorder.test_type_register_byproducts
|
|
||||||
msgid "Register By-products"
|
|
||||||
msgstr "按产品注册"
|
|
||||||
|
|
||||||
#. module: mrp_workorder
|
|
||||||
#: model:quality.point.test_type,name:mrp_workorder.test_type_print_label
|
|
||||||
msgid "Print label"
|
|
||||||
msgstr "打印标签"
|
|
||||||
@@ -138,7 +138,7 @@ if env.user.has_group('mrp.group_mrp_workorder_dependencies'):
|
|||||||
<button name="openMenuPopup" t-att-disabled="isBlocked" class="btn btn-secondary o_workorder_icon_btn fa fa-bars" type="workorder_event" title="menu"/>
|
<button name="openMenuPopup" t-att-disabled="isBlocked" class="btn btn-secondary o_workorder_icon_btn fa fa-bars" type="workorder_event" title="menu"/>
|
||||||
<span groups="mrp_workorder.group_mrp_wo_tablet_timer">
|
<span groups="mrp_workorder.group_mrp_wo_tablet_timer">
|
||||||
<button name="button_pending" type="object" class="btn btn-secondary" attrs="{'invisible': ['|', ('is_user_working', '=', False), ('working_state', '=', 'blocked')]}" barcode_trigger="pause" string="PAUSE"/>
|
<button name="button_pending" type="object" class="btn btn-secondary" attrs="{'invisible': ['|', ('is_user_working', '=', False), ('working_state', '=', 'blocked')]}" barcode_trigger="pause" string="PAUSE"/>
|
||||||
<button name="button_start" type="object" class="btn btn-warning" attrs="{'invisible': ['|', '|', ('is_user_working', '=', True), ('working_state', '=', 'blocked'), ('state', '=', ('done','rework', 'cancel'))]}" barcode_trigger="pause" string="CONTINUE"/>
|
<button name="button_start" type="object" class="btn btn-warning" attrs="{'invisible': ['|', '|', ('is_user_working', '=', True), ('working_state', '=', 'blocked'), ('state', '=', ('done', 'cancel'))]}" barcode_trigger="pause" string="CONTINUE"/>
|
||||||
<button name="button_unblock" type="object" class="btn btn-secondary btn-danger o_unblock" attrs="{'invisible': [('working_state', '!=', 'blocked')]}">Unblock</button>
|
<button name="button_unblock" type="object" class="btn btn-secondary btn-danger o_unblock" attrs="{'invisible': [('working_state', '!=', 'blocked')]}">Unblock</button>
|
||||||
<field name="duration" widget="mrp_timer" class="ms-1" readonly="1"/>
|
<field name="duration" widget="mrp_timer" class="ms-1" readonly="1"/>
|
||||||
</span>
|
</span>
|
||||||
|
|||||||
@@ -1050,13 +1050,3 @@ msgstr "工作中心故障"
|
|||||||
#: model:ir.model.fields,field_description:quality.field_quality_point_test_type__active
|
#: model:ir.model.fields,field_description:quality.field_quality_point_test_type__active
|
||||||
msgid "active"
|
msgid "active"
|
||||||
msgstr "有效"
|
msgstr "有效"
|
||||||
|
|
||||||
#. module: quality
|
|
||||||
#: model:quality.point.test_type,name:quality.test_type_instructions
|
|
||||||
msgid "Instructions"
|
|
||||||
msgstr "使用说明"
|
|
||||||
|
|
||||||
#. module: quality
|
|
||||||
#: model:quality.point.test_type,name:quality.test_type_picture
|
|
||||||
msgid "Take a Picture"
|
|
||||||
msgstr "照片"
|
|
||||||
@@ -15,7 +15,7 @@ class TestType(models.Model):
|
|||||||
_description = "Quality Control Test Type"
|
_description = "Quality Control Test Type"
|
||||||
|
|
||||||
# Used instead of selection field in order to hide a choice depending on the view.
|
# Used instead of selection field in order to hide a choice depending on the view.
|
||||||
name = fields.Char('Name', required=True,translate=True)
|
name = fields.Char('Name', required=True)
|
||||||
technical_name = fields.Char('Technical name', required=True)
|
technical_name = fields.Char('Technical name', required=True)
|
||||||
active = fields.Boolean('active', default=True)
|
active = fields.Boolean('active', default=True)
|
||||||
|
|
||||||
|
|||||||
@@ -4,4 +4,3 @@
|
|||||||
from . import models
|
from . import models
|
||||||
from . import wizard
|
from . import wizard
|
||||||
from . import report
|
from . import report
|
||||||
from . import controllers
|
|
||||||
|
|||||||
@@ -8,7 +8,7 @@
|
|||||||
'sequence': 120,
|
'sequence': 120,
|
||||||
'summary': 'Control the quality of your products',
|
'summary': 'Control the quality of your products',
|
||||||
'website': 'https://www.odoo.com/app/quality',
|
'website': 'https://www.odoo.com/app/quality',
|
||||||
'depends': ['quality', 'sf_manufacturing', 'base_import'],
|
'depends': ['quality'],
|
||||||
'description': """
|
'description': """
|
||||||
Quality Control
|
Quality Control
|
||||||
===============
|
===============
|
||||||
@@ -20,15 +20,12 @@ Quality Control
|
|||||||
""",
|
""",
|
||||||
'data': [
|
'data': [
|
||||||
'data/quality_control_data.xml',
|
'data/quality_control_data.xml',
|
||||||
'wizard/import_complex_model.xml',
|
|
||||||
'wizard/quality_wizard_view.xml',
|
|
||||||
'report/worksheet_custom_reports.xml',
|
'report/worksheet_custom_reports.xml',
|
||||||
'report/worksheet_custom_report_templates.xml',
|
'report/worksheet_custom_report_templates.xml',
|
||||||
'views/quality_views.xml',
|
'views/quality_views.xml',
|
||||||
'views/product_views.xml',
|
'views/product_views.xml',
|
||||||
'views/stock_move_views.xml',
|
'views/stock_move_views.xml',
|
||||||
'views/stock_picking_views.xml',
|
'views/stock_picking_views.xml',
|
||||||
'views/quality.check.measures.line.xml',
|
|
||||||
'wizard/quality_check_wizard_views.xml',
|
'wizard/quality_check_wizard_views.xml',
|
||||||
'security/ir.model.access.csv',
|
'security/ir.model.access.csv',
|
||||||
],
|
],
|
||||||
|
|||||||
@@ -1 +0,0 @@
|
|||||||
from . import main
|
|
||||||
@@ -1,120 +0,0 @@
|
|||||||
# -*- coding: utf-8 -*-
|
|
||||||
|
|
||||||
from odoo import http
|
|
||||||
from odoo.http import request, Response
|
|
||||||
import base64
|
|
||||||
import json
|
|
||||||
|
|
||||||
|
|
||||||
class QualityController(http.Controller):
|
|
||||||
|
|
||||||
@http.route(['/api/quality/report/download'], type='http', auth='public', csrf=False, website=False) # 移除 cors="*"
|
|
||||||
def get_quality_report(self, retrospect_ref=None, **kwargs):
|
|
||||||
"""获取质检报告的下载接口
|
|
||||||
|
|
||||||
Args:
|
|
||||||
retrospect_ref: 追溯码
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
直接返回文件下载响应
|
|
||||||
"""
|
|
||||||
try:
|
|
||||||
# 如果retrospect_ref为None,尝试从查询参数获取
|
|
||||||
if not retrospect_ref:
|
|
||||||
retrospect_ref = kwargs.get('retrospect_ref')
|
|
||||||
|
|
||||||
# 参数验证
|
|
||||||
if not retrospect_ref:
|
|
||||||
return self._json_response({
|
|
||||||
'status': 'error',
|
|
||||||
'message': '追溯码不能为空'
|
|
||||||
})
|
|
||||||
|
|
||||||
# 查找对应的质检单
|
|
||||||
quality_check = request.env['quality.check'].sudo().search([
|
|
||||||
('picking_id.retrospect_ref', '=', retrospect_ref),
|
|
||||||
('publish_status', '=', 'published') # 只返回已发布的报告
|
|
||||||
], limit=1)
|
|
||||||
|
|
||||||
if not quality_check:
|
|
||||||
return self._json_response({
|
|
||||||
'status': 'error',
|
|
||||||
'message': '未找到对应的质检报告或报告未发布'
|
|
||||||
})
|
|
||||||
|
|
||||||
if not quality_check.report_number_id:
|
|
||||||
return self._json_response({
|
|
||||||
'status': 'error',
|
|
||||||
'message': '质检报告文件不存在'
|
|
||||||
})
|
|
||||||
|
|
||||||
# 获取文件内容
|
|
||||||
document = quality_check.report_number_id
|
|
||||||
if not document.raw: # 检查文件内容是否存在
|
|
||||||
return self._json_response({
|
|
||||||
'status': 'error',
|
|
||||||
'message': '文件内容不存在'
|
|
||||||
})
|
|
||||||
|
|
||||||
# 构建文件名(确保有.pdf后缀)
|
|
||||||
filename = document.name
|
|
||||||
if not filename.lower().endswith('.pdf'):
|
|
||||||
filename = f"{filename}.pdf"
|
|
||||||
|
|
||||||
# 返回文件下载响应
|
|
||||||
return Response(
|
|
||||||
document.raw,
|
|
||||||
headers=[
|
|
||||||
('Content-Type', 'application/pdf'),
|
|
||||||
('Content-Disposition', f'attachment; filename="{filename}"'),
|
|
||||||
('Access-Control-Allow-Origin', '*'),
|
|
||||||
('Access-Control-Allow-Methods', 'GET, OPTIONS'),
|
|
||||||
('Access-Control-Allow-Headers', 'Content-Type, Authorization')
|
|
||||||
]
|
|
||||||
)
|
|
||||||
|
|
||||||
except Exception as e:
|
|
||||||
return self._json_response({
|
|
||||||
'status': 'error',
|
|
||||||
'message': f'系统错误: {str(e)}'
|
|
||||||
})
|
|
||||||
|
|
||||||
def _json_response(self, data):
|
|
||||||
"""返回JSON格式的响应"""
|
|
||||||
return Response(
|
|
||||||
json.dumps(data, ensure_ascii=False),
|
|
||||||
mimetype='application/json;charset=utf-8',
|
|
||||||
headers=[
|
|
||||||
('Access-Control-Allow-Origin', '*'),
|
|
||||||
('Access-Control-Allow-Methods', 'GET, OPTIONS'),
|
|
||||||
('Access-Control-Allow-Headers', 'Content-Type, Authorization')
|
|
||||||
]
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
class QualityReportController(http.Controller):
|
|
||||||
@http.route('/quality/report/<int:document_id>', type='http', auth='public')
|
|
||||||
def get_public_report(self, document_id, **kw):
|
|
||||||
"""提供公开访问PDF报告的控制器"""
|
|
||||||
document = request.env['documents.document'].sudo().browse(int(document_id))
|
|
||||||
|
|
||||||
# 安全检查:确保只有质检报告文档可以被访问
|
|
||||||
if document.exists() and document.res_model == 'quality.check':
|
|
||||||
# 获取PDF内容
|
|
||||||
pdf_content = document.raw
|
|
||||||
|
|
||||||
# 返回PDF内容
|
|
||||||
return request.make_response(
|
|
||||||
pdf_content,
|
|
||||||
headers=[
|
|
||||||
('Content-Type', 'application/pdf'),
|
|
||||||
('Content-Disposition', f'inline; filename={document.name}.pdf')
|
|
||||||
]
|
|
||||||
)
|
|
||||||
return request.not_found()
|
|
||||||
|
|
||||||
@http.route('/quality/report/not_published', type='http', auth='public')
|
|
||||||
def get_not_published_report(self, **kw):
|
|
||||||
"""提供未发布报告的控制器"""
|
|
||||||
return "报告尚未发布"
|
|
||||||
|
|
||||||
@@ -1185,14 +1185,3 @@ msgstr "请先进行质量检查!"
|
|||||||
#: model_terms:ir.ui.view,arch_db:quality_control.quality_alert_team_view_form
|
#: model_terms:ir.ui.view,arch_db:quality_control.quality_alert_team_view_form
|
||||||
msgid "e.g. The QA Masters"
|
msgid "e.g. The QA Masters"
|
||||||
msgstr "例如:QA大师"
|
msgstr "例如:QA大师"
|
||||||
|
|
||||||
|
|
||||||
#. module: quality_control
|
|
||||||
#: model:quality.point.test_type,name:quality_control.test_type_passfail
|
|
||||||
msgid "Pass - Fail"
|
|
||||||
msgstr "通过-失败"
|
|
||||||
|
|
||||||
#. module: quality_control
|
|
||||||
#: model:quality.point.test_type,name:quality_control.test_type_measure
|
|
||||||
msgid "Measure"
|
|
||||||
msgstr "测量"
|
|
||||||
@@ -6,4 +6,3 @@ from . import stock_move
|
|||||||
from . import stock_move_line
|
from . import stock_move_line
|
||||||
from . import stock_picking
|
from . import stock_picking
|
||||||
from . import stock_lot
|
from . import stock_lot
|
||||||
from . import product_category
|
|
||||||
@@ -1,32 +0,0 @@
|
|||||||
# -*- coding: utf-8 -*-
|
|
||||||
# Part of Odoo. See LICENSE file for full copyright and licensing details.
|
|
||||||
from math import sqrt
|
|
||||||
from dateutil.relativedelta import relativedelta
|
|
||||||
from datetime import datetime
|
|
||||||
|
|
||||||
import random
|
|
||||||
|
|
||||||
from odoo import api, models, fields, _
|
|
||||||
from odoo.tools import DEFAULT_SERVER_DATETIME_FORMAT, float_round
|
|
||||||
from odoo.osv.expression import OR
|
|
||||||
|
|
||||||
|
|
||||||
class ProductCategory(models.Model):
|
|
||||||
_inherit = 'product.category'
|
|
||||||
@api.model
|
|
||||||
def name_search(self, name='', args=None, operator='ilike', limit=100):
|
|
||||||
if args is None:
|
|
||||||
args = []
|
|
||||||
# 添加过滤条件,确保只返回名称为 'abc' 的记录
|
|
||||||
args += [('name', 'not in', ['Saleable', 'Expenses', 'Deliveries'])]
|
|
||||||
|
|
||||||
# 调用父类的 name_search 方法
|
|
||||||
return super(ProductCategory, self).name_search(name, args=args, operator=operator, limit=limit)
|
|
||||||
|
|
||||||
@api.model
|
|
||||||
def search(self, args, limit=100, offset=0, order=None, count=False):
|
|
||||||
# 添加过滤条件,确保只返回名称不在指定列表中的记录
|
|
||||||
args += [('name', 'not in', ['Saleable', 'Expenses', 'Deliveries'])]
|
|
||||||
|
|
||||||
# 调用父类的 search 方法
|
|
||||||
return super(ProductCategory, self).search(args, limit=limit, offset=offset, order=order, count=count)
|
|
||||||
@@ -7,15 +7,8 @@ from datetime import datetime
|
|||||||
import random
|
import random
|
||||||
|
|
||||||
from odoo import api, models, fields, _
|
from odoo import api, models, fields, _
|
||||||
from odoo.api import depends
|
|
||||||
from odoo.tools import DEFAULT_SERVER_DATETIME_FORMAT, float_round
|
from odoo.tools import DEFAULT_SERVER_DATETIME_FORMAT, float_round
|
||||||
from odoo.osv.expression import OR
|
from odoo.osv.expression import OR
|
||||||
from odoo.exceptions import UserError
|
|
||||||
from odoo.tools import image_data_uri
|
|
||||||
from base64 import b64encode
|
|
||||||
import requests
|
|
||||||
import json
|
|
||||||
import base64
|
|
||||||
|
|
||||||
|
|
||||||
class QualityPoint(models.Model):
|
class QualityPoint(models.Model):
|
||||||
@@ -40,8 +33,7 @@ class QualityPoint(models.Model):
|
|||||||
('day', 'Days'),
|
('day', 'Days'),
|
||||||
('week', 'Weeks'),
|
('week', 'Weeks'),
|
||||||
('month', 'Months')], default="day") # TDE RENAME ?
|
('month', 'Months')], default="day") # TDE RENAME ?
|
||||||
is_lot_tested_fractionally = fields.Boolean(string="Lot Tested Fractionally",
|
is_lot_tested_fractionally = fields.Boolean(string="Lot Tested Fractionally", help="Determines if only a fraction of the lot should be tested")
|
||||||
help="Determines if only a fraction of the lot should be tested")
|
|
||||||
testing_percentage_within_lot = fields.Float(help="Defines the percentage within a lot that should be tested")
|
testing_percentage_within_lot = fields.Float(help="Defines the percentage within a lot that should be tested")
|
||||||
norm = fields.Float('Norm', digits='Quality Tests') # TDE RENAME ?
|
norm = fields.Float('Norm', digits='Quality Tests') # TDE RENAME ?
|
||||||
tolerance_min = fields.Float('Min Tolerance', digits='Quality Tests')
|
tolerance_min = fields.Float('Min Tolerance', digits='Quality Tests')
|
||||||
@@ -70,7 +62,7 @@ class QualityPoint(models.Model):
|
|||||||
|
|
||||||
if n > 1:
|
if n > 1:
|
||||||
point.average = mean
|
point.average = mean
|
||||||
point.standard_deviation = sqrt(s / (n - 1))
|
point.standard_deviation = sqrt( s / ( n - 1))
|
||||||
elif n == 1:
|
elif n == 1:
|
||||||
point.average = mean
|
point.average = mean
|
||||||
point.standard_deviation = 0.0
|
point.standard_deviation = 0.0
|
||||||
@@ -101,7 +93,7 @@ class QualityPoint(models.Model):
|
|||||||
checks = self.env['quality.check'].search([
|
checks = self.env['quality.check'].search([
|
||||||
('point_id', '=', self.id),
|
('point_id', '=', self.id),
|
||||||
('create_date', '>=', date_previous.strftime(DEFAULT_SERVER_DATETIME_FORMAT))], limit=1)
|
('create_date', '>=', date_previous.strftime(DEFAULT_SERVER_DATETIME_FORMAT))], limit=1)
|
||||||
return not (bool(checks))
|
return not(bool(checks))
|
||||||
return super(QualityPoint, self).check_execute_now()
|
return super(QualityPoint, self).check_execute_now()
|
||||||
|
|
||||||
def _get_type_default_domain(self):
|
def _get_type_default_domain(self):
|
||||||
@@ -130,508 +122,6 @@ class QualityPoint(models.Model):
|
|||||||
|
|
||||||
class QualityCheck(models.Model):
|
class QualityCheck(models.Model):
|
||||||
_inherit = "quality.check"
|
_inherit = "quality.check"
|
||||||
part_name = fields.Char('零件名称', related='product_id.part_name', readonly=False, store=True)
|
|
||||||
part_number = fields.Char('零件图号', related='product_id.part_number', readonly=False, store=True)
|
|
||||||
material_name = fields.Char('材料名称', compute='_compute_material_name')
|
|
||||||
model_id = fields.Char('模型ID', related='product_id.model_id')
|
|
||||||
|
|
||||||
# # 总数量,值为调拨单_产品明细_数量
|
|
||||||
# total_qty = fields.Float('总数量', compute='_compute_total_qty', readonly=True)
|
|
||||||
# # 检验数
|
|
||||||
# check_qty = fields.Float('检验数', compute='_compute_check_qty', readonly=True)
|
|
||||||
# # 出厂检验报告编号
|
|
||||||
# report_number = fields.Char('出厂检验报告编号', compute='_compute_report_number', readonly=True)
|
|
||||||
# 总数量,值为调拨单_产品明细_数量
|
|
||||||
total_qty = fields.Char('总数量', compute='_compute_total_qty', store=True)
|
|
||||||
|
|
||||||
column_nums = fields.Integer('测量值列数', default=1)
|
|
||||||
|
|
||||||
@api.depends('picking_id')
|
|
||||||
def _compute_total_qty(self):
|
|
||||||
for record in self:
|
|
||||||
if record.picking_id:
|
|
||||||
total_qty = 0
|
|
||||||
for move in record.picking_id.move_ids_without_package:
|
|
||||||
if move.product_id == record.product_id:
|
|
||||||
total_qty = int(move.product_uom_qty)
|
|
||||||
record.total_qty = total_qty if total_qty > 0 else 0
|
|
||||||
else:
|
|
||||||
record.total_qty = 0
|
|
||||||
|
|
||||||
# 检验数
|
|
||||||
check_qty = fields.Integer('检验数', default=lambda self: self._get_default_check_qty())
|
|
||||||
|
|
||||||
def _get_default_check_qty(self):
|
|
||||||
"""根据条件设置检验数的默认值"""
|
|
||||||
# 这里需要使用_origin来获取已存储的记录,因为新记录在创建时可能还没有这些值
|
|
||||||
if self._origin:
|
|
||||||
if self._origin.measure_on == 'product' and self._origin.test_type_id.name == '出厂检验报告':
|
|
||||||
return ''
|
|
||||||
elif self._origin.measure_on == 'product':
|
|
||||||
return '1'
|
|
||||||
return ''
|
|
||||||
|
|
||||||
@api.onchange('test_type_id', 'measure_on')
|
|
||||||
def _onchange_check_qty(self):
|
|
||||||
"""当测试类型或测量对象变化时,更新检验数"""
|
|
||||||
if self.measure_on == 'product' and self.test_type_id.name == '出厂检验报告':
|
|
||||||
self.check_qty = 0
|
|
||||||
elif self.measure_on == 'product':
|
|
||||||
self.check_qty = 1
|
|
||||||
|
|
||||||
# 出厂检验报告编号
|
|
||||||
report_number_id = fields.Many2one('documents.document', string='出厂检验报告编号', readonly=True)
|
|
||||||
report_number_name = fields.Char('出厂检验报告编号名称', compute='_compute_report_number_name')
|
|
||||||
|
|
||||||
old_report_name = fields.Char('旧出厂检验报告编号', default='')
|
|
||||||
|
|
||||||
@api.depends('serial_number', 'part_number')
|
|
||||||
def _compute_report_number_name(self):
|
|
||||||
for record in self:
|
|
||||||
str_serial_number = '0' + str(record.serial_number) if record.serial_number < 10 else str(
|
|
||||||
record.serial_number)
|
|
||||||
str_part_number = record.part_number if record.part_number else ''
|
|
||||||
record.report_number_name = f'FQC{str_part_number}{str_serial_number}'
|
|
||||||
|
|
||||||
# 出厂检验报告、关联文档的数据
|
|
||||||
report_content = fields.Binary(string='出厂检验报告', related='report_number_id.datas')
|
|
||||||
|
|
||||||
is_out_check = fields.Boolean(string='是否出库检验', compute='_compute_is_out_check', readonly=True)
|
|
||||||
|
|
||||||
measure_line_ids = fields.One2many('quality.check.measure.line', 'check_id', string='测量明细')
|
|
||||||
|
|
||||||
categ_type = fields.Selection(string='产品的类别', related='product_id.categ_id.type', store=True)
|
|
||||||
|
|
||||||
report_result = fields.Selection([
|
|
||||||
('OK', 'OK'),
|
|
||||||
('NG', 'NG')
|
|
||||||
], string='出厂检验报告结果', default='OK')
|
|
||||||
measure_operator = fields.Many2one('res.users', string='操机员')
|
|
||||||
quality_manager = fields.Many2one('res.users', string='质检员', compute='_compute_quality_manager')
|
|
||||||
|
|
||||||
@api.depends('measure_line_ids')
|
|
||||||
def _compute_quality_manager(self):
|
|
||||||
for record in self:
|
|
||||||
if record.measure_line_ids:
|
|
||||||
record.quality_manager = record.env.user.id
|
|
||||||
else:
|
|
||||||
record.quality_manager = False
|
|
||||||
|
|
||||||
# 流水号(从1开始,最大99)
|
|
||||||
serial_number = fields.Integer('流水号', default=1, readonly=True)
|
|
||||||
|
|
||||||
# 发布历史
|
|
||||||
report_history_ids = fields.One2many('quality.check.report.history', 'check_id', string='发布历史')
|
|
||||||
|
|
||||||
# 发布状态
|
|
||||||
publish_status = fields.Selection([
|
|
||||||
('draft', '草稿'),
|
|
||||||
('published', '已发布'),
|
|
||||||
('canceled', '已撤销')
|
|
||||||
], string='发布状态', default='draft')
|
|
||||||
|
|
||||||
# 出厂检验报告是否已上传
|
|
||||||
is_factory_report_uploaded = fields.Boolean(string='出厂检验报告是否已上传', default=False)
|
|
||||||
|
|
||||||
def add_measure_line(self):
|
|
||||||
"""
|
|
||||||
新增测量值,如果测量值有5列了,则提示“最多只能有5列测量值”
|
|
||||||
"""
|
|
||||||
if self.column_nums >= 5:
|
|
||||||
raise UserError(_('最多只能有5列测量值'))
|
|
||||||
else:
|
|
||||||
for line in self.measure_line_ids:
|
|
||||||
field_name = f'measure_value{self.column_nums + 1}'
|
|
||||||
if hasattr(line, field_name):
|
|
||||||
line[field_name] = False
|
|
||||||
self.column_nums = self.column_nums + 1
|
|
||||||
|
|
||||||
def remove_measure_line(self):
|
|
||||||
"""
|
|
||||||
删除测量值
|
|
||||||
"""
|
|
||||||
if self.column_nums <= 1:
|
|
||||||
raise UserError(_('最少要有1列测量值'))
|
|
||||||
else:
|
|
||||||
for line in self.measure_line_ids:
|
|
||||||
field_name = f'measure_value{self.column_nums}'
|
|
||||||
if hasattr(line, field_name):
|
|
||||||
line[field_name] = False
|
|
||||||
self.column_nums = self.column_nums - 1
|
|
||||||
|
|
||||||
def upload_measure_line(self):
|
|
||||||
"""
|
|
||||||
上传测量值
|
|
||||||
"""
|
|
||||||
|
|
||||||
for record in self:
|
|
||||||
if not record.part_name or not record.part_number:
|
|
||||||
raise UserError(_('零件名称和零件图号均不能为空'))
|
|
||||||
|
|
||||||
# 如果验证通过,返回原动作
|
|
||||||
action = self.env.ref('quality_control.import_complex_model_wizard').read()[0]
|
|
||||||
action['context'] = {
|
|
||||||
'default_model_name': 'quality.check.measure.line',
|
|
||||||
'default_check_id': self.id,
|
|
||||||
}
|
|
||||||
return action
|
|
||||||
|
|
||||||
def do_preview(self):
|
|
||||||
"""
|
|
||||||
预览出厂检验报告
|
|
||||||
"""
|
|
||||||
pass
|
|
||||||
|
|
||||||
def do_publish(self):
|
|
||||||
"""发布出厂检验报告"""
|
|
||||||
self.ensure_one()
|
|
||||||
self._check_part_number()
|
|
||||||
self._check_measure_line()
|
|
||||||
self._check_check_qty_and_total_qty()
|
|
||||||
|
|
||||||
# 打开确认向导而不是直接发布
|
|
||||||
return {
|
|
||||||
'name': _('发布确认'),
|
|
||||||
'type': 'ir.actions.act_window',
|
|
||||||
'res_model': 'quality.check.publish.wizard',
|
|
||||||
'view_mode': 'form',
|
|
||||||
'target': 'new',
|
|
||||||
'context': {
|
|
||||||
'default_check_id': self.id,
|
|
||||||
'default_product_name': self.product_id.name,
|
|
||||||
'default_total_qty': self.total_qty,
|
|
||||||
'default_check_qty': self.check_qty,
|
|
||||||
'default_measure_count': self.column_nums,
|
|
||||||
'default_item_count': len(self.measure_line_ids),
|
|
||||||
'default_old_report_name': self.old_report_name,
|
|
||||||
'default_publish_status': self.publish_status,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
def _do_publish_implementation(self):
|
|
||||||
"""实际执行发布操作的方法"""
|
|
||||||
self.ensure_one()
|
|
||||||
|
|
||||||
# 1. 获取已发布的文档文件夹
|
|
||||||
workspace = self.env['documents.folder'].search(
|
|
||||||
[('parent_folder_id', '=', self.env.ref('sf_quality.documents_purchase_contracts_folder').id),
|
|
||||||
('name', '=', '已发布')], limit=1)
|
|
||||||
|
|
||||||
if self.serial_number > 99:
|
|
||||||
raise UserError(_('流水号不能大于99'))
|
|
||||||
|
|
||||||
# 2. 先创建空文档记录
|
|
||||||
doc_vals = {
|
|
||||||
'name': self.report_number_name,
|
|
||||||
'mimetype': 'application/pdf',
|
|
||||||
'res_id': self.id,
|
|
||||||
'folder_id': workspace.id,
|
|
||||||
'res_model': self._name,
|
|
||||||
}
|
|
||||||
|
|
||||||
doc = self.env['documents.document'].create(doc_vals)
|
|
||||||
|
|
||||||
# 3. 关联文档到质检记录
|
|
||||||
self.write({
|
|
||||||
'report_number_id': doc.id,
|
|
||||||
'quality_state': 'pass'
|
|
||||||
})
|
|
||||||
|
|
||||||
# 4. 获取报告动作并生成PDF(此时二维码将包含正确的文档ID)
|
|
||||||
report_action = self.env.ref('sf_quality.action_report_quality_inspection')
|
|
||||||
pdf_content, v = report_action._render_qweb_pdf(
|
|
||||||
report_ref=report_action.report_name,
|
|
||||||
res_ids=self.ids
|
|
||||||
)
|
|
||||||
|
|
||||||
# 5. 更新文档内容
|
|
||||||
doc.write({
|
|
||||||
'raw': pdf_content
|
|
||||||
})
|
|
||||||
|
|
||||||
# 6. 记录发布历史
|
|
||||||
self.env['quality.check.report.history'].create({
|
|
||||||
'check_id': self.id,
|
|
||||||
'report_number_id': doc.id,
|
|
||||||
'action': 'publish',
|
|
||||||
'operator': self.env.user.name,
|
|
||||||
'operation_time': datetime.now(),
|
|
||||||
'document_status': 'published',
|
|
||||||
'sequence': len(self.report_history_ids) + 1
|
|
||||||
})
|
|
||||||
|
|
||||||
# 7. 更新其他信息
|
|
||||||
self.serial_number += 1
|
|
||||||
|
|
||||||
if self.publish_status == 'canceled' and self.picking_id.state == 'done':
|
|
||||||
self.upload_factory_report()
|
|
||||||
|
|
||||||
self.write({
|
|
||||||
'publish_status': 'published',
|
|
||||||
})
|
|
||||||
|
|
||||||
return True
|
|
||||||
|
|
||||||
# 发布前检验零件图号、操机员、质检员
|
|
||||||
def _check_part_number(self):
|
|
||||||
if not self.part_number:
|
|
||||||
raise UserError(_('零件图号不能为空'))
|
|
||||||
if not self.measure_operator:
|
|
||||||
raise UserError(_('操机员不能为空'))
|
|
||||||
|
|
||||||
# 发布前校验明细行列均非空
|
|
||||||
def _check_measure_line(self):
|
|
||||||
for record in self:
|
|
||||||
if not record.measure_line_ids:
|
|
||||||
raise UserError(_('请先添加测量明细'))
|
|
||||||
for line in record.measure_line_ids:
|
|
||||||
if not line.measure_item:
|
|
||||||
raise UserError(_('有检测项目值为空'))
|
|
||||||
for i in range(1, record.column_nums + 1):
|
|
||||||
if not getattr(line, f'measure_value{i}'):
|
|
||||||
raise UserError(_('有测量值为空'))
|
|
||||||
|
|
||||||
# 发布前校验检验数与总数量、检验数与测量件数(即测量列数)
|
|
||||||
def _check_check_qty_and_total_qty(self):
|
|
||||||
for record in self:
|
|
||||||
if not record.check_qty:
|
|
||||||
raise UserError(_('请先输入检验数'))
|
|
||||||
if not record.total_qty:
|
|
||||||
raise UserError(_('总数量不能为空'))
|
|
||||||
if record.check_qty > int(record.total_qty):
|
|
||||||
raise UserError(_('检验数不可超过总数量'))
|
|
||||||
if record.column_nums > record.check_qty:
|
|
||||||
raise UserError(_('测量件数不可超过检验数'))
|
|
||||||
|
|
||||||
def do_cancel_publish(self):
|
|
||||||
"""
|
|
||||||
取消发布出厂检验报告(将当前质检单关联的出厂检验报告文档位置移动到废弃文件夹), 并记录发布历史
|
|
||||||
"""
|
|
||||||
self.ensure_one()
|
|
||||||
# 1. 获取已发布的文档文件夹
|
|
||||||
workspace = self.env['documents.folder'].search(
|
|
||||||
[('parent_folder_id', '=', self.env.ref('sf_quality.documents_purchase_contracts_folder').id),
|
|
||||||
('name', '=', '已发布')], limit=1)
|
|
||||||
# 2. 将当前质检单关联的出厂检验报告文档位置移动到废弃文件夹
|
|
||||||
self.report_number_id.write({
|
|
||||||
'folder_id': self.env.ref('sf_quality.documents_purchase_contracts_folder_canceled').id,
|
|
||||||
})
|
|
||||||
|
|
||||||
# 3. 记录发布历史
|
|
||||||
self.env['quality.check.report.history'].create({
|
|
||||||
'check_id': self.id,
|
|
||||||
'report_number_id': self.report_number_id.id,
|
|
||||||
'action': 'cancel_publish',
|
|
||||||
'operator': self.env.user.name,
|
|
||||||
'operation_time': datetime.now(),
|
|
||||||
'document_status': 'canceled',
|
|
||||||
'sequence': len(self.report_history_ids) + 1
|
|
||||||
})
|
|
||||||
|
|
||||||
self.write({
|
|
||||||
'old_report_name': self.report_number_id.name
|
|
||||||
})
|
|
||||||
|
|
||||||
# 3. 更新发布状态
|
|
||||||
self.write({
|
|
||||||
'publish_status': 'canceled',
|
|
||||||
'report_number_id': False,
|
|
||||||
'quality_state': 'none'
|
|
||||||
})
|
|
||||||
|
|
||||||
if self.is_factory_report_uploaded:
|
|
||||||
# 4. 删除加工订单明细中的出厂检验报告
|
|
||||||
self.delete_factory_report()
|
|
||||||
|
|
||||||
return True
|
|
||||||
|
|
||||||
def do_re_publish(self):
|
|
||||||
"""
|
|
||||||
重新发布出厂检验报告,参考发布规则
|
|
||||||
"""
|
|
||||||
return self.do_publish()
|
|
||||||
|
|
||||||
def generate_qr_code(self):
|
|
||||||
"""生成二维码URL"""
|
|
||||||
self.ensure_one()
|
|
||||||
base_url = self.env['ir.config_parameter'].sudo().get_param('web.base.url')
|
|
||||||
return image_data_uri(
|
|
||||||
b64encode(self.env['ir.actions.report'].barcode(
|
|
||||||
'QR', base_url + '/#/index/publicPay?order_id=' + str(self.id) + '&source=%2Findex%2Fmyorder',
|
|
||||||
width=140, height=140)
|
|
||||||
)
|
|
||||||
)
|
|
||||||
|
|
||||||
def get_latest_report_attachment(self, check_id):
|
|
||||||
"""获取指定质检记录的最新报告附件,并删除旧的报告附件"""
|
|
||||||
# 查找特定质检记录的所有附件
|
|
||||||
attachments = self.env['ir.attachment'].search([
|
|
||||||
('res_model', '=', 'quality.check'),
|
|
||||||
('res_id', '=', check_id),
|
|
||||||
('name', 'like', 'QC-QC') # 根据您的命名规则调整
|
|
||||||
], order='create_date DESC') # 按创建日期降序排序
|
|
||||||
|
|
||||||
# # 如果附件数量大于1,则删除除最新报告外的其他报告附件
|
|
||||||
# if len(attachments) > 1:
|
|
||||||
# for attachment in attachments[1:]:
|
|
||||||
# attachment.unlink()
|
|
||||||
|
|
||||||
# 返回最新的附件(如果存在)
|
|
||||||
return attachments and attachments[0] or False
|
|
||||||
|
|
||||||
def get_report_url(self):
|
|
||||||
"""生成报告访问URL"""
|
|
||||||
self.ensure_one()
|
|
||||||
base_url = self.env['ir.config_parameter'].sudo().get_param('web.base.url')
|
|
||||||
if self.report_number_id:
|
|
||||||
print(f"{base_url}/quality/report/{self.report_number_id.id}")
|
|
||||||
return f"{base_url}/quality/report/{self.report_number_id.id}"
|
|
||||||
else:
|
|
||||||
return f"{base_url}/quality/report/not_published"
|
|
||||||
|
|
||||||
def upload_factory_report(self):
|
|
||||||
"""
|
|
||||||
上传出厂检验报告到加工订单明细中
|
|
||||||
将当前质检单的出厂检验报告上传到对应的加工订单明细中
|
|
||||||
"""
|
|
||||||
self.ensure_one()
|
|
||||||
if not self.report_content:
|
|
||||||
raise UserError(_('当前质检单没有出厂检验报告,请先发布报告'))
|
|
||||||
|
|
||||||
if not self.product_id.model_name:
|
|
||||||
raise UserError(_('产品模型名称为空'))
|
|
||||||
|
|
||||||
if not self.picking_id or not self.picking_id.origin:
|
|
||||||
raise UserError(_('无法找到相关的调拨单或来源单据'))
|
|
||||||
|
|
||||||
# 获取订单号(从调拨单的来源字段获取)
|
|
||||||
order_ref = self.picking_id.retrospect_ref
|
|
||||||
|
|
||||||
try:
|
|
||||||
# 准备请求数据
|
|
||||||
payload = {
|
|
||||||
"order_ref": order_ref,
|
|
||||||
"model_name": self.product_id.model_name,
|
|
||||||
"report_file": self.report_content.decode('utf-8') if isinstance(self.report_content,
|
|
||||||
bytes) else self.report_content
|
|
||||||
}
|
|
||||||
|
|
||||||
# 将Python字典转换为JSON字符串
|
|
||||||
json_data = json.dumps(payload)
|
|
||||||
|
|
||||||
# 获取服务器URL
|
|
||||||
base_url = self.env['ir.config_parameter'].sudo().get_param('bfm_url_new')
|
|
||||||
api_url = f"{base_url}/api/report/create"
|
|
||||||
|
|
||||||
# 设置请求头
|
|
||||||
headers = {
|
|
||||||
'Content-Type': 'application/json',
|
|
||||||
}
|
|
||||||
|
|
||||||
# 发送POST请求
|
|
||||||
response = requests.post(api_url, data=json_data, headers=headers)
|
|
||||||
|
|
||||||
# 处理响应
|
|
||||||
if response.status_code == 200:
|
|
||||||
result = response.json()
|
|
||||||
if result.get('success'):
|
|
||||||
# 上传成功,显示成功消息
|
|
||||||
self.is_factory_report_uploaded = True
|
|
||||||
return {
|
|
||||||
'type': 'ir.actions.client',
|
|
||||||
'tag': 'display_notification',
|
|
||||||
'params': {
|
|
||||||
'title': _('上传成功'),
|
|
||||||
'message': _('出厂检验报告已成功上传到加工订单明细'),
|
|
||||||
'type': 'success',
|
|
||||||
'sticky': False,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else:
|
|
||||||
# API返回失败信息
|
|
||||||
raise UserError(_('上传失败: %s') % result.get('message', '未知错误'))
|
|
||||||
else:
|
|
||||||
# HTTP请求失败
|
|
||||||
raise UserError(_('请求失败,状态码: %s') % response.status_code)
|
|
||||||
|
|
||||||
except Exception as e:
|
|
||||||
raise UserError(_('上传过程中发生错误: %s') % str(e))
|
|
||||||
|
|
||||||
def delete_factory_report(self):
|
|
||||||
"""
|
|
||||||
删除加工订单明细中的出厂检验报告
|
|
||||||
"""
|
|
||||||
# 获取订单号(从调拨单的来源字段获取)
|
|
||||||
order_ref = self.picking_id.retrospect_ref
|
|
||||||
|
|
||||||
if not order_ref:
|
|
||||||
raise UserError(_('无法找到相关的调拨单或来源单据'))
|
|
||||||
|
|
||||||
if not self.product_id.model_name:
|
|
||||||
raise UserError(_('产品模型名称为空'))
|
|
||||||
|
|
||||||
try:
|
|
||||||
# 准备请求数据
|
|
||||||
payload = {
|
|
||||||
"order_ref": order_ref,
|
|
||||||
"model_name": self.product_id.model_name
|
|
||||||
}
|
|
||||||
|
|
||||||
# 将Python字典转换为JSON字符串
|
|
||||||
json_data = json.dumps(payload)
|
|
||||||
|
|
||||||
# 获取服务器URL
|
|
||||||
base_url = self.env['ir.config_parameter'].sudo().get_param('bfm_url_new')
|
|
||||||
api_url = f"{base_url}/api/report/delete"
|
|
||||||
|
|
||||||
# 设置请求头
|
|
||||||
headers = {
|
|
||||||
'Content-Type': 'application/json',
|
|
||||||
}
|
|
||||||
|
|
||||||
# 发送POST请求
|
|
||||||
response = requests.post(api_url, data=json_data, headers=headers)
|
|
||||||
|
|
||||||
# 处理响应
|
|
||||||
if response.status_code == 200:
|
|
||||||
result = response.json()
|
|
||||||
if result.get('success'):
|
|
||||||
# 删除成功,显示成功消息
|
|
||||||
self.is_factory_report_uploaded = False
|
|
||||||
return {
|
|
||||||
'type': 'ir.actions.client',
|
|
||||||
'tag': 'display_notification',
|
|
||||||
'params': {
|
|
||||||
'title': _('删除成功'),
|
|
||||||
'message': _('出厂检验报告已成功删除'),
|
|
||||||
'type': 'success',
|
|
||||||
'sticky': False,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else:
|
|
||||||
# API返回失败信息
|
|
||||||
raise UserError(_('删除失败: %s') % result.get('message', '未知错误'))
|
|
||||||
else:
|
|
||||||
# HTTP请求失败
|
|
||||||
raise UserError(_('请求失败,状态码: %s') % response.status_code)
|
|
||||||
|
|
||||||
except Exception as e:
|
|
||||||
raise UserError(_('删除过程中发生错误: %s') % str(e))
|
|
||||||
|
|
||||||
@depends('product_id')
|
|
||||||
def _compute_material_name(self):
|
|
||||||
for record in self:
|
|
||||||
materials_id_name = record.product_id.materials_id.name if record.product_id.materials_id else ''
|
|
||||||
materials_type_name = record.product_id.materials_type_id.name if record.product_id.materials_type_id else ''
|
|
||||||
record.material_name = materials_id_name + ' ' + materials_type_name
|
|
||||||
|
|
||||||
@depends('test_type_id')
|
|
||||||
def _compute_is_out_check(self):
|
|
||||||
for record in self:
|
|
||||||
if record.test_type_id.name == '出厂检验报告':
|
|
||||||
record.is_out_check = True
|
|
||||||
else:
|
|
||||||
record.is_out_check = False
|
|
||||||
|
|
||||||
failure_message = fields.Html(related='point_id.failure_message', readonly=True)
|
failure_message = fields.Html(related='point_id.failure_message', readonly=True)
|
||||||
measure = fields.Float('Measure', default=0.0, digits='Quality Tests', tracking=True)
|
measure = fields.Float('Measure', default=0.0, digits='Quality Tests', tracking=True)
|
||||||
@@ -644,8 +134,7 @@ class QualityCheck(models.Model):
|
|||||||
tolerance_max = fields.Float('Max Tolerance', related='point_id.tolerance_max', readonly=True)
|
tolerance_max = fields.Float('Max Tolerance', related='point_id.tolerance_max', readonly=True)
|
||||||
warning_message = fields.Text(compute='_compute_warning_message')
|
warning_message = fields.Text(compute='_compute_warning_message')
|
||||||
norm_unit = fields.Char(related='point_id.norm_unit', readonly=True)
|
norm_unit = fields.Char(related='point_id.norm_unit', readonly=True)
|
||||||
qty_to_test = fields.Float(compute="_compute_qty_to_test", string="Quantity to Test",
|
qty_to_test = fields.Float(compute="_compute_qty_to_test", string="Quantity to Test", help="Quantity of product to test within the lot")
|
||||||
help="Quantity of product to test within the lot")
|
|
||||||
qty_tested = fields.Float(string="Quantity Tested", help="Quantity of product tested within the lot")
|
qty_tested = fields.Float(string="Quantity Tested", help="Quantity of product tested within the lot")
|
||||||
measure_on = fields.Selection([
|
measure_on = fields.Selection([
|
||||||
('operation', 'Operation'),
|
('operation', 'Operation'),
|
||||||
@@ -654,8 +143,7 @@ class QualityCheck(models.Model):
|
|||||||
help="""Operation = One quality check is requested at the operation level.
|
help="""Operation = One quality check is requested at the operation level.
|
||||||
Product = A quality check is requested per product.
|
Product = A quality check is requested per product.
|
||||||
Quantity = A quality check is requested for each new product quantity registered, with partial quantity checks also possible.""")
|
Quantity = A quality check is requested for each new product quantity registered, with partial quantity checks also possible.""")
|
||||||
move_line_id = fields.Many2one('stock.move.line', 'Stock Move Line', check_company=True,
|
move_line_id = fields.Many2one('stock.move.line', 'Stock Move Line', check_company=True, help="In case of Quality Check by Quantity, Move Line on which the Quality Check applies")
|
||||||
help="In case of Quality Check by Quantity, Move Line on which the Quality Check applies")
|
|
||||||
lot_name = fields.Char('Lot/Serial Number Name')
|
lot_name = fields.Char('Lot/Serial Number Name')
|
||||||
lot_line_id = fields.Many2one('stock.lot', store=True, compute='_compute_lot_line_id')
|
lot_line_id = fields.Many2one('stock.lot', store=True, compute='_compute_lot_line_id')
|
||||||
qty_line = fields.Float(compute='_compute_qty_line', string="Quantity")
|
qty_line = fields.Float(compute='_compute_qty_line', string="Quantity")
|
||||||
@@ -664,34 +152,6 @@ class QualityCheck(models.Model):
|
|||||||
is_lot_tested_fractionally = fields.Boolean(related='point_id.is_lot_tested_fractionally')
|
is_lot_tested_fractionally = fields.Boolean(related='point_id.is_lot_tested_fractionally')
|
||||||
testing_percentage_within_lot = fields.Float(related="point_id.testing_percentage_within_lot")
|
testing_percentage_within_lot = fields.Float(related="point_id.testing_percentage_within_lot")
|
||||||
product_tracking = fields.Selection(related='product_id.tracking')
|
product_tracking = fields.Selection(related='product_id.tracking')
|
||||||
quality_check_type = fields.Selection([
|
|
||||||
('采购入库检', '采购入库检'),
|
|
||||||
('客供料入库检', '客供料入库检'),
|
|
||||||
('退货入库检', '退货入库检'),
|
|
||||||
('生产入库检', '生产入库检'),
|
|
||||||
('外协入库检', '外协入库检'),
|
|
||||||
('成品发货检', '成品发货检'),
|
|
||||||
('工序外协发货检', '工序外协发货检'),
|
|
||||||
('委外坯料发货检', '委外坯料发货检')], string='类型', compute='_compute_quality_check_type', store=True)
|
|
||||||
|
|
||||||
@api.depends('picking_id')
|
|
||||||
def _compute_quality_check_type(self):
|
|
||||||
for check in self:
|
|
||||||
if check.picking_id:
|
|
||||||
picking_type = check.picking_id.picking_type_id.sequence_code
|
|
||||||
type_mapping = {
|
|
||||||
'IN': '采购入库检',
|
|
||||||
'DL': '客供料入库检',
|
|
||||||
'RET': '退货入库检',
|
|
||||||
'SFP': '生产入库检',
|
|
||||||
'OCIN': '外协入库检',
|
|
||||||
'OUT': '成品发货检',
|
|
||||||
'OCOUT': '工序外协发货检',
|
|
||||||
'RES': '委外坯料发货检',
|
|
||||||
}
|
|
||||||
check.quality_check_type = type_mapping.get(picking_type, False)
|
|
||||||
else:
|
|
||||||
check.quality_check_type = False
|
|
||||||
|
|
||||||
@api.depends('measure_success')
|
@api.depends('measure_success')
|
||||||
def _compute_warning_message(self):
|
def _compute_warning_message(self):
|
||||||
@@ -736,9 +196,7 @@ class QualityCheck(models.Model):
|
|||||||
def _compute_qty_to_test(self):
|
def _compute_qty_to_test(self):
|
||||||
for qc in self:
|
for qc in self:
|
||||||
if qc.is_lot_tested_fractionally:
|
if qc.is_lot_tested_fractionally:
|
||||||
rounding = qc.product_id.uom_id.rounding if qc.product_id.uom_id else 0.01
|
qc.qty_to_test = float_round(qc.qty_line * qc.testing_percentage_within_lot / 100, precision_rounding=self.product_id.uom_id.rounding, rounding_method="UP")
|
||||||
qc.qty_to_test = float_round(qc.qty_line * qc.testing_percentage_within_lot / 100,
|
|
||||||
precision_rounding=rounding, rounding_method="UP")
|
|
||||||
else:
|
else:
|
||||||
qc.qty_to_test = qc.qty_line
|
qc.qty_to_test = qc.qty_line
|
||||||
|
|
||||||
@@ -836,19 +294,6 @@ class QualityAlert(models.Model):
|
|||||||
_inherit = "quality.alert"
|
_inherit = "quality.alert"
|
||||||
|
|
||||||
title = fields.Char('Title')
|
title = fields.Char('Title')
|
||||||
part_number = fields.Char(string='零件图号', compute='_compute_part_info', store=True)
|
|
||||||
part_name = fields.Char(string='零件名称', compute='_compute_part_info', store=True)
|
|
||||||
|
|
||||||
@api.depends('product_id', 'picking_id')
|
|
||||||
def _compute_part_info(self):
|
|
||||||
for alert in self:
|
|
||||||
if alert.product_tmpl_id.categ_id.name == '成品':
|
|
||||||
alert.part_number = alert.product_id.part_number
|
|
||||||
alert.part_name = alert.product_id.part_name
|
|
||||||
elif alert.product_id.categ_id.name == '坯料':
|
|
||||||
if alert.picking_id.move_ids_without_package:
|
|
||||||
alert.part_number = alert.picking_id.move_ids_without_package[0].part_number
|
|
||||||
alert.part_name = alert.picking_id.move_ids_without_package[0].part_name
|
|
||||||
|
|
||||||
def action_see_check(self):
|
def action_see_check(self):
|
||||||
return {
|
return {
|
||||||
@@ -893,8 +338,7 @@ class QualityAlert(models.Model):
|
|||||||
class ProductTemplate(models.Model):
|
class ProductTemplate(models.Model):
|
||||||
_inherit = "product.template"
|
_inherit = "product.template"
|
||||||
|
|
||||||
quality_control_point_qty = fields.Integer(compute='_compute_quality_check_qty',
|
quality_control_point_qty = fields.Integer(compute='_compute_quality_check_qty', groups='quality.group_quality_user')
|
||||||
groups='quality.group_quality_user')
|
|
||||||
quality_pass_qty = fields.Integer(compute='_compute_quality_check_qty', groups='quality.group_quality_user')
|
quality_pass_qty = fields.Integer(compute='_compute_quality_check_qty', groups='quality.group_quality_user')
|
||||||
quality_fail_qty = fields.Integer(compute='_compute_quality_check_qty', groups='quality.group_quality_user')
|
quality_fail_qty = fields.Integer(compute='_compute_quality_check_qty', groups='quality.group_quality_user')
|
||||||
|
|
||||||
@@ -902,16 +346,14 @@ class ProductTemplate(models.Model):
|
|||||||
def _compute_quality_check_qty(self):
|
def _compute_quality_check_qty(self):
|
||||||
for product_tmpl in self:
|
for product_tmpl in self:
|
||||||
product_tmpl.quality_fail_qty, product_tmpl.quality_pass_qty = product_tmpl.product_variant_ids._count_quality_checks()
|
product_tmpl.quality_fail_qty, product_tmpl.quality_pass_qty = product_tmpl.product_variant_ids._count_quality_checks()
|
||||||
product_tmpl.quality_control_point_qty = product_tmpl.with_context(
|
product_tmpl.quality_control_point_qty = product_tmpl.with_context(active_test=product_tmpl.active).product_variant_ids._count_quality_points()
|
||||||
active_test=product_tmpl.active).product_variant_ids._count_quality_points()
|
|
||||||
|
|
||||||
def action_see_quality_control_points(self):
|
def action_see_quality_control_points(self):
|
||||||
self.ensure_one()
|
self.ensure_one()
|
||||||
action = self.env["ir.actions.actions"]._for_xml_id("quality_control.quality_point_action")
|
action = self.env["ir.actions.actions"]._for_xml_id("quality_control.quality_point_action")
|
||||||
action['context'] = dict(self.env.context, default_product_ids=self.product_variant_ids.ids)
|
action['context'] = dict(self.env.context, default_product_ids=self.product_variant_ids.ids)
|
||||||
|
|
||||||
domain_in_products_or_categs = ['|', ('product_ids', 'in', self.product_variant_ids.ids),
|
domain_in_products_or_categs = ['|', ('product_ids', 'in', self.product_variant_ids.ids), ('product_category_ids', 'parent_of', self.categ_id.ids)]
|
||||||
('product_category_ids', 'parent_of', self.categ_id.ids)]
|
|
||||||
domain_no_products_and_categs = [('product_ids', '=', False), ('product_category_ids', '=', False)]
|
domain_no_products_and_categs = [('product_ids', '=', False), ('product_category_ids', '=', False)]
|
||||||
action['domain'] = OR([domain_in_products_or_categs, domain_no_products_and_categs])
|
action['domain'] = OR([domain_in_products_or_categs, domain_no_products_and_categs])
|
||||||
return action
|
return action
|
||||||
@@ -922,10 +364,10 @@ class ProductTemplate(models.Model):
|
|||||||
action['context'] = dict(self.env.context, default_product_id=self.product_variant_id.id, create=False)
|
action['context'] = dict(self.env.context, default_product_id=self.product_variant_id.id, create=False)
|
||||||
action['domain'] = [
|
action['domain'] = [
|
||||||
'|',
|
'|',
|
||||||
('product_id', 'in', self.product_variant_ids.ids),
|
('product_id', 'in', self.product_variant_ids.ids),
|
||||||
'&',
|
'&',
|
||||||
('measure_on', '=', 'operation'),
|
('measure_on', '=', 'operation'),
|
||||||
('picking_id.move_ids.product_tmpl_id', '=', self.id),
|
('picking_id.move_ids.product_tmpl_id', '=', self.id),
|
||||||
]
|
]
|
||||||
return action
|
return action
|
||||||
|
|
||||||
@@ -933,8 +375,7 @@ class ProductTemplate(models.Model):
|
|||||||
class ProductProduct(models.Model):
|
class ProductProduct(models.Model):
|
||||||
_inherit = "product.product"
|
_inherit = "product.product"
|
||||||
|
|
||||||
quality_control_point_qty = fields.Integer(compute='_compute_quality_check_qty',
|
quality_control_point_qty = fields.Integer(compute='_compute_quality_check_qty', groups='quality.group_quality_user')
|
||||||
groups='quality.group_quality_user')
|
|
||||||
quality_pass_qty = fields.Integer(compute='_compute_quality_check_qty', groups='quality.group_quality_user')
|
quality_pass_qty = fields.Integer(compute='_compute_quality_check_qty', groups='quality.group_quality_user')
|
||||||
quality_fail_qty = fields.Integer(compute='_compute_quality_check_qty', groups='quality.group_quality_user')
|
quality_fail_qty = fields.Integer(compute='_compute_quality_check_qty', groups='quality.group_quality_user')
|
||||||
|
|
||||||
@@ -948,10 +389,10 @@ class ProductProduct(models.Model):
|
|||||||
quality_pass_qty = 0
|
quality_pass_qty = 0
|
||||||
domain = [
|
domain = [
|
||||||
'|',
|
'|',
|
||||||
('product_id', 'in', self.ids),
|
('product_id', 'in', self.ids),
|
||||||
'&',
|
'&',
|
||||||
('measure_on', '=', 'operation'),
|
('measure_on', '=', 'operation'),
|
||||||
('picking_id.move_ids.product_id', 'in', self.ids),
|
('picking_id.move_ids.product_id', 'in', self.ids),
|
||||||
('company_id', '=', self.env.company.id),
|
('company_id', '=', self.env.company.id),
|
||||||
('quality_state', '!=', 'none')
|
('quality_state', '!=', 'none')
|
||||||
]
|
]
|
||||||
@@ -975,8 +416,7 @@ class ProductProduct(models.Model):
|
|||||||
_, where_clause, where_clause_args = query.get_sql()
|
_, where_clause, where_clause_args = query.get_sql()
|
||||||
additional_where_clause = self._additional_quality_point_where_clause()
|
additional_where_clause = self._additional_quality_point_where_clause()
|
||||||
where_clause += additional_where_clause
|
where_clause += additional_where_clause
|
||||||
parent_category_ids = [int(parent_id) for parent_id in
|
parent_category_ids = [int(parent_id) for parent_id in self.categ_id.parent_path.split('/')[:-1]] if self.categ_id else []
|
||||||
self.categ_id.parent_path.split('/')[:-1]] if self.categ_id else []
|
|
||||||
|
|
||||||
self.env.cr.execute("""
|
self.env.cr.execute("""
|
||||||
SELECT COUNT(*)
|
SELECT COUNT(*)
|
||||||
@@ -997,7 +437,7 @@ class ProductProduct(models.Model):
|
|||||||
)
|
)
|
||||||
)
|
)
|
||||||
""" % (where_clause,), where_clause_args + [self.ids, parent_category_ids]
|
""" % (where_clause,), where_clause_args + [self.ids, parent_category_ids]
|
||||||
)
|
)
|
||||||
return self.env.cr.fetchone()[0]
|
return self.env.cr.fetchone()[0]
|
||||||
|
|
||||||
def action_see_quality_control_points(self):
|
def action_see_quality_control_points(self):
|
||||||
@@ -1005,8 +445,7 @@ class ProductProduct(models.Model):
|
|||||||
action = self.product_tmpl_id.action_see_quality_control_points()
|
action = self.product_tmpl_id.action_see_quality_control_points()
|
||||||
action['context'].update(default_product_ids=self.ids)
|
action['context'].update(default_product_ids=self.ids)
|
||||||
|
|
||||||
domain_in_products_or_categs = ['|', ('product_ids', 'in', self.ids),
|
domain_in_products_or_categs = ['|', ('product_ids', 'in', self.ids), ('product_category_ids', 'parent_of', self.categ_id.ids)]
|
||||||
('product_category_ids', 'parent_of', self.categ_id.ids)]
|
|
||||||
domain_no_products_and_categs = [('product_ids', '=', False), ('product_category_ids', '=', False)]
|
domain_no_products_and_categs = [('product_ids', '=', False), ('product_category_ids', '=', False)]
|
||||||
action['domain'] = OR([domain_in_products_or_categs, domain_no_products_and_categs])
|
action['domain'] = OR([domain_in_products_or_categs, domain_no_products_and_categs])
|
||||||
return action
|
return action
|
||||||
@@ -1017,75 +456,12 @@ class ProductProduct(models.Model):
|
|||||||
action['context'] = dict(self.env.context, default_product_id=self.id, create=False)
|
action['context'] = dict(self.env.context, default_product_id=self.id, create=False)
|
||||||
action['domain'] = [
|
action['domain'] = [
|
||||||
'|',
|
'|',
|
||||||
('product_id', '=', self.id),
|
('product_id', '=', self.id),
|
||||||
'&',
|
'&',
|
||||||
('measure_on', '=', 'operation'),
|
('measure_on', '=', 'operation'),
|
||||||
('picking_id.move_ids.product_id', '=', self.id),
|
('picking_id.move_ids.product_id', '=', self.id),
|
||||||
]
|
]
|
||||||
return action
|
return action
|
||||||
|
|
||||||
def _additional_quality_point_where_clause(self):
|
def _additional_quality_point_where_clause(self):
|
||||||
return ""
|
return ""
|
||||||
|
|
||||||
|
|
||||||
class QualityCheckMeasureLine(models.Model):
|
|
||||||
_name = 'quality.check.measure.line'
|
|
||||||
_description = '质检测量明细'
|
|
||||||
_order = 'sequence, id'
|
|
||||||
|
|
||||||
sequence = fields.Integer('序号')
|
|
||||||
|
|
||||||
check_id = fields.Many2one('quality.check', string='质检单', required=True, ondelete='cascade')
|
|
||||||
|
|
||||||
# 基本信息
|
|
||||||
product_name = fields.Char('产品名称', related='check_id.product_id.name', readonly=True)
|
|
||||||
drawing_no = fields.Char('图号')
|
|
||||||
measure_item = fields.Char('检测项目')
|
|
||||||
|
|
||||||
# 测量值
|
|
||||||
measure_value1 = fields.Char('测量值1')
|
|
||||||
measure_value2 = fields.Char('测量值2')
|
|
||||||
measure_value3 = fields.Char('测量值3')
|
|
||||||
measure_value4 = fields.Char('测量值4')
|
|
||||||
measure_value5 = fields.Char('测量值5')
|
|
||||||
|
|
||||||
# # 展示列数
|
|
||||||
# column_nums = fields.Integer('列数', related='check_id.column_nums')
|
|
||||||
|
|
||||||
# 判定结果
|
|
||||||
measure_result = fields.Selection([
|
|
||||||
('OK', '合格'),
|
|
||||||
('NG', '不合格')
|
|
||||||
], string='判定', default='OK')
|
|
||||||
|
|
||||||
remark = fields.Char('备注')
|
|
||||||
|
|
||||||
def del_measure_value(self):
|
|
||||||
self.ensure_one()
|
|
||||||
self.sudo().unlink()
|
|
||||||
|
|
||||||
|
|
||||||
# 增加出厂检验报告发布历史
|
|
||||||
class QualityCheckReportHistory(models.Model):
|
|
||||||
_name = 'quality.check.report.history'
|
|
||||||
_description = '出厂检验报告发布历史'
|
|
||||||
|
|
||||||
check_id = fields.Many2one('quality.check', string='质检单', required=True, ondelete='cascade')
|
|
||||||
report_number_id = fields.Many2one('documents.document', string='报告编号', readonly=True)
|
|
||||||
|
|
||||||
sequence = fields.Integer('序号')
|
|
||||||
# 操作(发布、撤销发布、重新发布)
|
|
||||||
action = fields.Selection([
|
|
||||||
('publish', '发布'),
|
|
||||||
('cancel_publish', '撤销发布'),
|
|
||||||
('re_publish', '重新发布')
|
|
||||||
], string='操作')
|
|
||||||
# 操作人
|
|
||||||
operator = fields.Char('操作人')
|
|
||||||
# 操作时间
|
|
||||||
operation_time = fields.Datetime('操作时间')
|
|
||||||
# 文档状态(已发布、废弃)
|
|
||||||
document_status = fields.Selection([
|
|
||||||
('published', '已发布'),
|
|
||||||
('canceled', '废弃')
|
|
||||||
], string='操作后文档状态')
|
|
||||||
|
|||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user