Compare commits
469 Commits
feature/67
...
master
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
fe8df494f9 | ||
|
|
32cd68e15f | ||
|
|
f0e47371ed | ||
|
|
6dde814acc | ||
|
|
71ab241e94 | ||
|
|
0a13acbb68 | ||
|
|
4b026535f8 | ||
|
|
99ac89f995 | ||
|
|
35b1d648c3 | ||
|
|
fe3492ceb5 | ||
|
|
4274b9fe99 | ||
|
|
4ddabdefa1 | ||
|
|
364127beb3 | ||
|
|
109ea8729d | ||
|
|
32de726164 | ||
|
|
79f89f068b | ||
|
|
52189cff72 | ||
|
|
aadccce47e | ||
|
|
abd88fd721 | ||
|
|
2c2fa87719 | ||
|
|
7d8b7048a8 | ||
|
|
e7b312fb22 | ||
|
|
bc641bde3a | ||
|
|
d73ffab9b5 | ||
|
|
4b1c0bc3b5 | ||
|
|
45730e2dd4 | ||
|
|
8903f32cf8 | ||
|
|
8bd1c8a095 | ||
|
|
abf9a0997b | ||
|
|
aec2d1c516 | ||
|
|
127efb67d7 | ||
|
|
78edf1f7eb | ||
|
|
e6e13a5970 | ||
|
|
55cc4906ef | ||
|
|
427b548939 | ||
|
|
8c9db7e8c8 | ||
|
|
990f73ea4e | ||
|
|
d21e0c7fd9 | ||
|
|
b4ed65a75c | ||
|
|
53ba2ec589 | ||
|
|
566bc0e1e2 | ||
|
|
5f216684f0 | ||
|
|
e86bd59a43 | ||
|
|
e55625e4b6 | ||
|
|
790dd3dd05 | ||
|
|
84846fb3da | ||
|
|
f45a46255b | ||
|
|
fe5b005e45 | ||
|
|
fc33280509 | ||
|
|
60ed04cf77 | ||
|
|
3439b47b0c | ||
|
|
33c5d648b5 | ||
|
|
c7b50d9706 | ||
|
|
05caed1d91 | ||
|
|
8b66fda899 | ||
|
|
1456d4303b | ||
|
|
9d2f23de7c | ||
|
|
9393a48e70 | ||
|
|
749f44d6df | ||
|
|
4f000a6be4 | ||
|
|
7af7079a5c | ||
|
|
376eb9e56f | ||
|
|
531bc2e090 | ||
|
|
bc3ab21d75 | ||
|
|
d1811ea24b | ||
|
|
b426d5d505 | ||
|
|
1dfa22900d | ||
|
|
0eeebf437a | ||
|
|
397d4f29a1 | ||
|
|
33205c5d29 | ||
|
|
568f3e4f30 | ||
|
|
8c43fc2b76 | ||
|
|
8960d3d07c | ||
|
|
95b5c86242 | ||
|
|
61034c3424 | ||
|
|
c2000aa9c5 | ||
|
|
e29456bbf7 | ||
|
|
10e995ec7f | ||
|
|
a2f8dc6cec | ||
|
|
fec095ca6b | ||
|
|
aed33dbb35 | ||
|
|
dbe8c95558 | ||
|
|
5c7e6e969f | ||
|
|
d70b757487 | ||
|
|
fb3bb8f1c0 | ||
|
|
9123aeaee8 | ||
|
|
e3af0bea3c | ||
|
|
af1bc021d6 | ||
|
|
2febc369bb | ||
|
|
6ee1c5f9a9 | ||
|
|
5ef8023169 | ||
|
|
fa03b562a2 | ||
|
|
5f55c954d1 | ||
|
|
e0ca13b5b7 | ||
|
|
dba38f4d37 | ||
|
|
ceb7a02209 | ||
|
|
1811dbf0fd | ||
|
|
1c1d1a74ad | ||
|
|
347019d7ee | ||
|
|
7d46d00fd7 | ||
|
|
5a22402e7a | ||
|
|
f53e34aeb4 | ||
|
|
e145e8a3a4 | ||
|
|
0ef6fe73f3 | ||
|
|
ffad4b7995 | ||
|
|
2c7fbd3aef | ||
|
|
72b8d33a3e | ||
|
|
6517d2bd12 | ||
|
|
2c97287218 | ||
|
|
012ff120b4 | ||
|
|
960f05090c | ||
|
|
6321e7ef23 | ||
|
|
69d200973b | ||
|
|
448a2cd277 | ||
|
|
5a071188cc | ||
|
|
b8894609a9 | ||
|
|
3f8fd6da62 | ||
|
|
154a17657c | ||
|
|
62ead52f00 | ||
|
|
23d6e38b24 | ||
|
|
bdc23afc56 | ||
|
|
b8043b3ad2 | ||
|
|
8841d800ea | ||
|
|
920e96ffc6 | ||
|
|
d52f5fa841 | ||
|
|
37c5c9d498 | ||
|
|
6867d7e4ce | ||
|
|
9cbd311fec | ||
|
|
0e753b1c85 | ||
|
|
48316c55b7 | ||
|
|
b33ba9c354 | ||
|
|
e0559e9887 | ||
|
|
39a25bb6c8 | ||
|
|
796e9b0cef | ||
|
|
e129c08426 | ||
|
|
c6b47bd68d | ||
|
|
126d60f8d7 | ||
|
|
4225a8fe1b | ||
|
|
a13a79f41f | ||
|
|
a828c823dd | ||
|
|
9cf2bac9c6 | ||
|
|
1926375d58 | ||
|
|
23dd88b7ba | ||
|
|
f164488e48 | ||
|
|
25b53794bb | ||
|
|
484fab85be | ||
|
|
6ed5de6400 | ||
|
|
8224f36567 | ||
|
|
2449b92bc8 | ||
|
|
03ec94d223 | ||
|
|
d26e6edd31 | ||
|
|
2ea24f2049 | ||
|
|
b11b6ef283 | ||
|
|
b1a04f8f44 | ||
|
|
0b5415dc47 | ||
|
|
59569806e6 | ||
|
|
4e1be6f4d5 | ||
|
|
cdf8fbb12a | ||
|
|
5d0f094da7 | ||
|
|
865d2216af | ||
|
|
95cb5251dc | ||
|
|
1d01e3ad2e | ||
|
|
c8fe7504c7 | ||
|
|
222efc57c2 | ||
|
|
735d5c659d | ||
|
|
50d188b737 | ||
|
|
af3ea0f702 | ||
|
|
8bf3b68cee | ||
|
|
2766bc7d34 | ||
|
|
1082384d00 | ||
|
|
25aab1576d | ||
|
|
87891b45ef | ||
|
|
b2cfdd8d78 | ||
|
|
540b7bcbea | ||
|
|
c6cb1d367d | ||
|
|
001900bd65 | ||
|
|
0204e0e24f | ||
|
|
2b3a2dd21c | ||
|
|
57acad4716 | ||
|
|
9f97c82a46 | ||
|
|
7cafddd345 | ||
|
|
c796697a8e | ||
|
|
ee523e9aac | ||
|
|
87fdc7bf74 | ||
|
|
fc41f30244 | ||
|
|
fbcd8c57c5 | ||
|
|
e7d84e9df2 | ||
|
|
b7642d1e0f | ||
|
|
05ffbdcc78 | ||
|
|
4b8d00ec1d | ||
|
|
d2dbf4f986 | ||
|
|
b0f2fe6a8e | ||
|
|
edfd59468f | ||
|
|
d7f04b61b5 | ||
|
|
573b50da68 | ||
|
|
2e0dfc5b02 | ||
|
|
18cdc39719 | ||
|
|
de951b1b45 | ||
|
|
b8cebe07fe | ||
|
|
14fa88da01 | ||
|
|
162814411f | ||
|
|
1bdb81f5f7 | ||
|
|
49e4c88d83 | ||
|
|
db81114a07 | ||
|
|
e019383187 | ||
|
|
4a09148b53 | ||
|
|
007f39f137 | ||
|
|
ab139daf02 | ||
|
|
24d83b70d2 | ||
|
|
307510e1ab | ||
|
|
9b7222961c | ||
|
|
3304398c4c | ||
|
|
074e59cee4 | ||
|
|
7ab8ab47ac | ||
|
|
9b01254b3c | ||
|
|
3a89ebff60 | ||
|
|
d0ee2a6564 | ||
|
|
257d4a3b0a | ||
|
|
53a67d7c76 | ||
|
|
40fe1f15a2 | ||
|
|
30a6e5bf2e | ||
|
|
f4babfcd24 | ||
|
|
794ea0cbb0 | ||
|
|
619285608d | ||
|
|
433d5d63b7 | ||
|
|
38109028d4 | ||
|
|
cf16a9dd59 | ||
|
|
4fcbeb30cf | ||
|
|
be8fbca9aa | ||
|
|
b393951968 | ||
|
|
6add565a98 | ||
|
|
1b0dd96b40 | ||
|
|
89f8718fe1 | ||
|
|
a5243970d5 | ||
|
|
7941c1981c | ||
|
|
f31e25b3b1 | ||
|
|
ae7e49e307 | ||
|
|
51c8287bbc | ||
|
|
942d6661f2 | ||
|
|
51ae598aac | ||
|
|
83229c9ab1 | ||
|
|
e62f933ca4 | ||
|
|
1f8e118965 | ||
|
|
e46e6dfc2a | ||
|
|
2c52372b0a | ||
|
|
50f8bf5ab1 | ||
|
|
5cf3d399f4 | ||
|
|
e9fc78186e | ||
|
|
a2b2faaa95 | ||
|
|
4c7208784f | ||
|
|
981569170c | ||
|
|
6bf666ac18 | ||
|
|
3a9cd3f39d | ||
|
|
45b62abcbe | ||
|
|
87740dbee3 | ||
|
|
a323acf7fc | ||
|
|
c23715a1ef | ||
|
|
896c1ad3a7 | ||
|
|
1856a1a4ef | ||
|
|
8bd09cddf0 | ||
|
|
52579673de | ||
|
|
31cd131993 | ||
|
|
9f94a4e06f | ||
|
|
e66c6b1e1b | ||
|
|
e97d0af941 | ||
|
|
65122d38d5 | ||
|
|
b626cbe217 | ||
|
|
4b443e65f6 | ||
|
|
14700d6bf0 | ||
|
|
a3c0fd3ccf | ||
|
|
a29265f17d | ||
|
|
0821ed021a | ||
|
|
ac4883db66 | ||
|
|
5706aa0052 | ||
|
|
f780e4f7ce | ||
|
|
e6d8ebb7b3 | ||
|
|
3663e04b34 | ||
|
|
c2e4085b50 | ||
|
|
33c881b12f | ||
|
|
c80e12d731 | ||
|
|
5446ef18e2 | ||
|
|
9c4d545915 | ||
|
|
f6d8cb6267 | ||
|
|
c898e02860 | ||
|
|
5477582a69 | ||
|
|
9cb22d810e | ||
|
|
cab6b6fa2a | ||
|
|
35bf954529 | ||
|
|
ceb38aa483 | ||
|
|
11ecad5ef2 | ||
|
|
8249d1427f | ||
|
|
94bcfc0543 | ||
|
|
b4d31c7c4b | ||
|
|
61bcd72a41 | ||
|
|
d7f7bb9a57 | ||
|
|
ee87e1dacf | ||
|
|
2f6c41c999 | ||
|
|
d0d4db1555 | ||
|
|
62cbb4b796 | ||
|
|
f040406002 | ||
|
|
bfff4ac440 | ||
|
|
a97386c37c | ||
|
|
18ae46207a | ||
|
|
bacddd2ad8 | ||
|
|
dd5794899d | ||
|
|
e5b730b2ef | ||
|
|
aea158de41 | ||
|
|
a933a0ffea | ||
|
|
7575424760 | ||
|
|
6c2eb40e6a | ||
|
|
f10f595fa4 | ||
|
|
6d1de42d76 | ||
|
|
5dc16c039c | ||
|
|
c416cdbeed | ||
|
|
18c7b22319 | ||
|
|
b5339046b9 | ||
|
|
e0ba222382 | ||
|
|
58b00e6442 | ||
|
|
9182dbfb5d | ||
|
|
27516844af | ||
|
|
99237445ac | ||
|
|
9349ca91d3 | ||
|
|
51c517145b | ||
|
|
c55f3d77bf | ||
|
|
95716c2e3e | ||
|
|
5f72519dc2 | ||
|
|
c24bba3137 | ||
|
|
01bb6fd0aa | ||
|
|
bf4add6b78 | ||
|
|
51a8964b89 | ||
|
|
7d986fe139 | ||
|
|
fffbfc21c2 | ||
|
|
6451bfbc42 | ||
|
|
5aa848de53 | ||
|
|
efc4ae31c9 | ||
|
|
0863238819 | ||
|
|
4f181e5eba | ||
|
|
2bf43ae9a1 | ||
|
|
d98d04d4ed | ||
|
|
602d6678bc | ||
|
|
8fd0c4e1f1 | ||
|
|
514fd79c3e | ||
|
|
95c25ac7b8 | ||
|
|
21d052e222 | ||
|
|
95e2c2db0d | ||
|
|
17a29b7b29 | ||
|
|
dd745423a1 | ||
|
|
a534e5f400 | ||
|
|
4dc7b5857e | ||
|
|
dc679c46cc | ||
|
|
8ccf6cc365 | ||
|
|
f8457ae66b | ||
|
|
12c8641f2e | ||
|
|
f42938f668 | ||
|
|
a856c5cbf7 | ||
|
|
6411e79904 | ||
|
|
946f08c479 | ||
|
|
4a198639ec | ||
|
|
234812bb40 | ||
|
|
dd43e31c3c | ||
|
|
2f5b0281c3 | ||
|
|
d4cf2a9d17 | ||
|
|
ecf5dcf2f2 | ||
|
|
848e8a5fa8 | ||
|
|
cc38383e32 | ||
|
|
39de4e5ea1 | ||
|
|
8b6c904dae | ||
|
|
a63f2d28f6 | ||
|
|
08812f169e | ||
|
|
ce79016bef | ||
|
|
fef960f7e8 | ||
|
|
425c9fb64b | ||
|
|
fc9a58c0c3 | ||
|
|
ed90ad34e6 | ||
|
|
5662094ec4 | ||
|
|
404c56e134 | ||
|
|
9ee614aa10 | ||
|
|
57789dc5a5 | ||
|
|
52d436909b | ||
|
|
3a760a66e1 | ||
|
|
72415d633c | ||
|
|
5c67a8c190 | ||
|
|
46ba682848 | ||
|
|
6b38062e87 | ||
|
|
0945754736 | ||
|
|
644ff967e5 | ||
|
|
5f79d2038c | ||
|
|
defd779279 | ||
|
|
e2e820267e | ||
|
|
94f179a6d6 | ||
|
|
bf9f4c1276 | ||
|
|
51a633594f | ||
|
|
7d7c7b0fcf | ||
|
|
d88ac22b7c | ||
|
|
1f4e1c11c8 | ||
|
|
9f1beb4013 | ||
|
|
f864466987 | ||
|
|
9cf70cc54c | ||
|
|
82bd50cb97 | ||
|
|
4bce26721d | ||
|
|
3fb4e7c413 | ||
|
|
a7ab8679f4 | ||
|
|
ca9a91e30a | ||
|
|
314d738412 | ||
|
|
699e03ccda | ||
|
|
8f0ade7b43 | ||
|
|
50bc8786e8 | ||
|
|
0777e63bc7 | ||
|
|
128bebf338 | ||
|
|
7a71077aa7 | ||
|
|
10a1d43a17 | ||
|
|
87d351e9e9 | ||
|
|
d2daae1a8f | ||
|
|
5997c24895 | ||
|
|
df53989f22 | ||
|
|
9bab687080 | ||
|
|
a5ac8b8b84 | ||
|
|
2cde398e11 | ||
|
|
88026fea5d | ||
|
|
443a21a0cc | ||
|
|
e14646a6fc | ||
|
|
6a920be6d1 | ||
|
|
3811079a7f | ||
|
|
ad8e0b6af0 | ||
|
|
04cb910803 | ||
|
|
42292818af | ||
|
|
bcafd9cf38 | ||
|
|
12ebd87f1d | ||
|
|
bdef852b98 | ||
|
|
1d5fb747d4 | ||
|
|
8116e4f97d | ||
|
|
e3e5fcc378 | ||
|
|
879b5492db | ||
|
|
27b9a4f982 | ||
|
|
94007bae2b | ||
|
|
bf92028027 | ||
|
|
2b47e566d3 | ||
|
|
5aa2f1aa18 | ||
|
|
b7128ba81a | ||
|
|
49546f9d08 | ||
|
|
6959bd9a09 | ||
|
|
3a2babf2d5 | ||
|
|
d7d094c84d | ||
|
|
a06e24583d | ||
|
|
0cbd830901 | ||
|
|
4b29def105 | ||
|
|
582abb3f2e | ||
|
|
40137ba69c | ||
|
|
804f6a82b4 | ||
|
|
d16d47dfbe | ||
|
|
41cf9d5474 | ||
|
|
59aa6b4f10 | ||
|
|
a759106fdc | ||
|
|
8bb101c6b2 | ||
|
|
f02044b513 | ||
|
|
3d937b85c9 | ||
|
|
5a61b3b459 | ||
|
|
afccb5ee6a | ||
|
|
2b0648d9bc | ||
|
|
8ea3487044 | ||
|
|
b24ed5fe4c | ||
|
|
b801b265c3 | ||
|
|
27a67167fe | ||
|
|
8fa9534b4e | ||
|
|
db745e46b6 | ||
|
|
f598b6c71c | ||
|
|
dccb0b3fb0 | ||
|
|
ac09794b10 | ||
|
|
bff0ff9401 |
@@ -119,13 +119,24 @@ patch(ListRenderer.prototype, 'jikimo_frontend.ListRenderer', {
|
|||||||
this.listherHeaderBodyNum()
|
this.listherHeaderBodyNum()
|
||||||
})
|
})
|
||||||
const treeModifiers = this.getFieldModifiers(this.props.archInfo.__rawArch);
|
const treeModifiers = this.getFieldModifiers(this.props.archInfo.__rawArch);
|
||||||
|
|
||||||
if(treeModifiers) {
|
if(treeModifiers) {
|
||||||
this.props.merge_key = treeModifiers.merge_key;
|
if(treeModifiers.merge_fields) {
|
||||||
this.props.merge_fields = treeModifiers.merge_fields.split(',');
|
this.props.merge_key = treeModifiers.merge_key;
|
||||||
const data = this.setColumns(this.props.merge_key);
|
this.props.merge_fields = treeModifiers.merge_fields.split(',');
|
||||||
owl.onMounted(() => {
|
const data = this.setColumns(this.props.merge_key);
|
||||||
this.mergeColumns(this.props.merge_fields, data)
|
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);
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -521,11 +521,6 @@ div:has(.o_required_modifier) > label::before {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// 设置表格横向滚动
|
|
||||||
.o_list_renderer.o_renderer {
|
|
||||||
max-width: 100%;
|
|
||||||
overflow-x: auto;
|
|
||||||
}
|
|
||||||
|
|
||||||
// 设置表单页面label文本不换行
|
// 设置表单页面label文本不换行
|
||||||
.o_form_view .o_group .o_wrap_label .o_form_label {
|
.o_form_view .o_group .o_wrap_label .o_form_label {
|
||||||
|
|||||||
2
jikimo_printing/__init__.py
Normal file
2
jikimo_printing/__init__.py
Normal file
@@ -0,0 +1,2 @@
|
|||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
from . import models
|
||||||
18
jikimo_printing/__manifest__.py
Normal file
18
jikimo_printing/__manifest__.py
Normal file
@@ -0,0 +1,18 @@
|
|||||||
|
# -*- 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',
|
||||||
|
}
|
||||||
5
jikimo_printing/models/__init__.py
Normal file
5
jikimo_printing/models/__init__.py
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
from . import jikimo_printing
|
||||||
|
from . import maintenance_printing
|
||||||
|
from . import workorder_printing
|
||||||
|
|
||||||
87
jikimo_printing/models/jikimo_printing.py
Normal file
87
jikimo_printing/models/jikimo_printing.py
Normal file
@@ -0,0 +1,87 @@
|
|||||||
|
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
|
||||||
69
jikimo_printing/models/maintenance_printing.py
Normal file
69
jikimo_printing/models/maintenance_printing.py
Normal file
@@ -0,0 +1,69 @@
|
|||||||
|
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,8 +1,6 @@
|
|||||||
|
|
||||||
import logging
|
import logging
|
||||||
from io import BytesIO
|
|
||||||
from odoo import models, fields, api
|
from odoo import models, fields, api
|
||||||
from odoo.exceptions import UserError
|
|
||||||
|
|
||||||
_logger = logging.getLogger(__name__)
|
_logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
@@ -21,12 +19,8 @@ class MrpWorkorder(models.Model):
|
|||||||
if pdf_data:
|
if pdf_data:
|
||||||
try:
|
try:
|
||||||
# 执行打印
|
# 执行打印
|
||||||
printer = self.env['printing.printer'].get_default()
|
self.env['jikimo.printing'].print_pdf(pdf_data)
|
||||||
printer.print_document(report=None, content = pdf_data, doc_format='pdf')
|
|
||||||
|
|
||||||
wo.production_id.product_id.is_print_program = True
|
wo.production_id.product_id.is_print_program = True
|
||||||
_logger.info(f"工单 {wo.name} 的PDF已成功打印")
|
|
||||||
|
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
_logger.error(f"工单 {wo.name} 的PDF打印失败: {str(e)}")
|
_logger.error(f"工单 {wo.name} 的PDF打印失败: {str(e)}")
|
||||||
|
|
||||||
19
jikimo_printing/views/maintenance_views.xml
Normal file
19
jikimo_printing/views/maintenance_views.xml
Normal file
@@ -0,0 +1,19 @@
|
|||||||
|
<?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>
|
||||||
|
|
||||||
|
|
||||||
@@ -8,16 +8,21 @@
|
|||||||
'category': 'purchase',
|
'category': 'purchase',
|
||||||
'depends': ['sf_manufacturing', 'purchase_request'],
|
'depends': ['sf_manufacturing', 'purchase_request'],
|
||||||
'data': [
|
'data': [
|
||||||
|
'security/ir.model.access.csv',
|
||||||
'views/sale_order_view.xml',
|
'views/sale_order_view.xml',
|
||||||
'views/mrp_production.xml',
|
'views/mrp_production.xml',
|
||||||
'views/purchase_request_view.xml',
|
'views/purchase_request_view.xml',
|
||||||
'wizard/purchase_request_line_make_purchase_order_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': {
|
'assets': {
|
||||||
# 'web.assets_backend': [
|
'web.assets_backend': [
|
||||||
# 'jikimo_purchase_request/static/src/**/*'
|
'jikimo_purchase_request/static/src/**/*'
|
||||||
# ],
|
],
|
||||||
# },
|
},
|
||||||
'application': True,
|
'application': True,
|
||||||
'installable': True,
|
'installable': True,
|
||||||
'auto_install': False,
|
'auto_install': False,
|
||||||
|
|||||||
@@ -410,7 +410,7 @@ msgstr "显示名称"
|
|||||||
#: model_terms:ir.ui.view,arch_db:purchase_request.view_purchase_request_form
|
#: model_terms:ir.ui.view,arch_db:purchase_request.view_purchase_request_form
|
||||||
#: model_terms:ir.ui.view,arch_db:purchase_request.view_purchase_request_search
|
#: model_terms:ir.ui.view,arch_db:purchase_request.view_purchase_request_search
|
||||||
msgid "Done"
|
msgid "Done"
|
||||||
msgstr "完成"
|
msgstr "关闭"
|
||||||
|
|
||||||
#. module: purchase_request
|
#. module: purchase_request
|
||||||
#: model:ir.model.fields,field_description:purchase_request.field_purchase_request_line__move_dest_ids
|
#: model:ir.model.fields,field_description:purchase_request.field_purchase_request_line__move_dest_ids
|
||||||
@@ -1043,7 +1043,7 @@ msgstr "询价单"
|
|||||||
#. module: purchase_request
|
#. module: purchase_request
|
||||||
#: model:ir.model.fields,field_description:purchase_request.field_purchase_request_line__purchased_qty
|
#: model:ir.model.fields,field_description:purchase_request.field_purchase_request_line__purchased_qty
|
||||||
msgid "RFQ/PO Qty"
|
msgid "RFQ/PO Qty"
|
||||||
msgstr ""
|
msgstr "已订购数"
|
||||||
|
|
||||||
#. module: purchase_request
|
#. module: purchase_request
|
||||||
#. odoo-python
|
#. odoo-python
|
||||||
|
|||||||
@@ -5,3 +5,5 @@ from . import sale_order
|
|||||||
from . import mrp_production
|
from . import mrp_production
|
||||||
from . import purchase_order
|
from . import purchase_order
|
||||||
from . import stock_rule
|
from . import stock_rule
|
||||||
|
from . import stock_picking
|
||||||
|
from . import product_product
|
||||||
|
|||||||
@@ -9,18 +9,22 @@ class MrpProduction(models.Model):
|
|||||||
@api.depends('state')
|
@api.depends('state')
|
||||||
def _compute_pr_mp_count(self):
|
def _compute_pr_mp_count(self):
|
||||||
for item in self:
|
for item in self:
|
||||||
pr_ids = self.env['purchase.request'].sudo().search([('origin', 'like', item.name), ('is_subcontract', '!=', 'True')])
|
if item.product_id.is_customer_provided:
|
||||||
if pr_ids:
|
|
||||||
item.pr_mp_count = len(pr_ids)
|
|
||||||
else:
|
|
||||||
item.pr_mp_count = 0
|
item.pr_mp_count = 0
|
||||||
|
else:
|
||||||
|
pr_ids = item._get_purchase_request()
|
||||||
|
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):
|
def action_view_pr_mp(self):
|
||||||
"""
|
"""
|
||||||
采购请求
|
采购请求
|
||||||
"""
|
"""
|
||||||
self.ensure_one()
|
self.ensure_one()
|
||||||
pr_ids = self.env['purchase.request'].sudo().search([('origin', 'like', self.name),('is_subcontract', '!=', True)])
|
|
||||||
|
# 由于采购申请合并了所有销售订单行的采购,所以不区分产品
|
||||||
|
pr_ids = self._get_purchase_request()
|
||||||
|
|
||||||
action = {
|
action = {
|
||||||
'res_model': 'purchase.request',
|
'res_model': 'purchase.request',
|
||||||
'type': 'ir.actions.act_window',
|
'type': 'ir.actions.act_window',
|
||||||
@@ -33,7 +37,16 @@ class MrpProduction(models.Model):
|
|||||||
else:
|
else:
|
||||||
action.update({
|
action.update({
|
||||||
'name': _("从 %s生成采购请求单", self.name),
|
'name': _("从 %s生成采购请求单", self.name),
|
||||||
'domain': [('id', 'in', pr_ids)],
|
'domain': [('id', 'in', pr_ids.ids)],
|
||||||
'view_mode': 'tree,form',
|
'view_mode': 'tree,form',
|
||||||
})
|
})
|
||||||
return action
|
return action
|
||||||
|
|
||||||
|
def _get_purchase_request(self):
|
||||||
|
"""获取跟制造订单相关的采购申请单(根据采购申请单行项目的产品匹配)"""
|
||||||
|
mrp_names = self.env['mrp.production'].search([('origin', '=', self.origin)]).mapped('name')
|
||||||
|
pr_ids = self.env['purchase.request'].sudo().search([('origin', 'in', mrp_names)])
|
||||||
|
product_list = self.product_id._get_product_include_bom()
|
||||||
|
pr_line_ids = pr_ids.line_ids.filtered(lambda l: l.product_id in product_list)
|
||||||
|
return pr_line_ids.mapped('request_id')
|
||||||
|
|
||||||
17
jikimo_purchase_request/models/product_product.py
Normal file
17
jikimo_purchase_request/models/product_product.py
Normal file
@@ -0,0 +1,17 @@
|
|||||||
|
from odoo import models
|
||||||
|
|
||||||
|
|
||||||
|
class ProductProduct(models.Model):
|
||||||
|
_inherit = 'product.product'
|
||||||
|
|
||||||
|
|
||||||
|
def _get_product_include_bom(self):
|
||||||
|
"""获取产品列表(包括所有bom)"""
|
||||||
|
self.ensure_one()
|
||||||
|
product_list = [self]
|
||||||
|
bom_ids = self.bom_ids
|
||||||
|
while (bom_ids):
|
||||||
|
bom_product_ids = bom_ids.bom_line_ids.mapped('product_id')
|
||||||
|
product_list.append(bom_product_ids)
|
||||||
|
bom_ids = bom_product_ids.bom_ids
|
||||||
|
return product_list
|
||||||
@@ -1,4 +1,5 @@
|
|||||||
from odoo import api, fields, models, _
|
from odoo import api, fields, models, _
|
||||||
|
from odoo.tools import float_compare
|
||||||
|
|
||||||
|
|
||||||
class PurchaseOrder(models.Model):
|
class PurchaseOrder(models.Model):
|
||||||
@@ -13,4 +14,43 @@ class PurchaseOrder(models.Model):
|
|||||||
('done', '完成'),
|
('done', '完成'),
|
||||||
('cancel', '取消'),
|
('cancel', '取消'),
|
||||||
('rejected', '已驳回')
|
('rejected', '已驳回')
|
||||||
], string='Status', readonly=True, index=True, copy=False, default='draft', tracking=True)
|
], 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,13 +1,15 @@
|
|||||||
import re
|
import re
|
||||||
import ast
|
import ast
|
||||||
from odoo import models, fields, api
|
from odoo import models, fields, api, _
|
||||||
|
from itertools import groupby
|
||||||
|
from odoo.tools import float_compare
|
||||||
|
|
||||||
|
|
||||||
class PurchaseRequest(models.Model):
|
class PurchaseRequest(models.Model):
|
||||||
_inherit = 'purchase.request'
|
_inherit = 'purchase.request'
|
||||||
_description = '采购申请'
|
_description = '采购申请'
|
||||||
|
|
||||||
# 为state添加取消状态
|
# 为state添加取消状态
|
||||||
state = fields.Selection(
|
state = fields.Selection(
|
||||||
selection_add=[('cancel', '已取消')],
|
selection_add=[('cancel', '已取消')],
|
||||||
ondelete={'cancel': 'set default'} # 添加 ondelete 策略
|
ondelete={'cancel': 'set default'} # 添加 ondelete 策略
|
||||||
@@ -29,6 +31,57 @@ class PurchaseRequest(models.Model):
|
|||||||
action['context'] = origin_context
|
action['context'] = origin_context
|
||||||
return action
|
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:
|
||||||
|
if line_item.state == 'purchase':
|
||||||
|
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):
|
class PurchaseRequestLine(models.Model):
|
||||||
_inherit = 'purchase.request.line'
|
_inherit = 'purchase.request.line'
|
||||||
_description = '采购申请明细'
|
_description = '采购申请明细'
|
||||||
@@ -47,6 +100,20 @@ class PurchaseRequestLine(models.Model):
|
|||||||
('outsourcing', "委外加工"),
|
('outsourcing', "委外加工"),
|
||||||
], string='供货方式', compute='_compute_supply_method', store=True)
|
], 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')
|
@api.depends('origin')
|
||||||
def _compute_supply_method(self):
|
def _compute_supply_method(self):
|
||||||
for prl in self:
|
for prl in self:
|
||||||
@@ -79,10 +146,12 @@ class PurchaseRequestLine(models.Model):
|
|||||||
continue
|
continue
|
||||||
if record.product_id.categ_id.name == '坯料':
|
if record.product_id.categ_id.name == '坯料':
|
||||||
product_name = ''
|
product_name = ''
|
||||||
match = re.search(r'(S\d{5}-\d)', record.product_id.name)
|
match = re.search(r'(S\d{5}-\d+)', record.product_id.name)
|
||||||
# 如果匹配成功,提取结果
|
# 如果匹配成功,提取结果
|
||||||
if match:
|
if match:
|
||||||
product_name = match.group(0)
|
product_name = match.group(0)
|
||||||
|
else:
|
||||||
|
product_name = record.product_id.name
|
||||||
sale_order_name = ''
|
sale_order_name = ''
|
||||||
match_sale = re.search(r'S(\d+)', record.product_id.name)
|
match_sale = re.search(r'S(\d+)', record.product_id.name)
|
||||||
if match_sale:
|
if match_sale:
|
||||||
@@ -98,3 +167,36 @@ class PurchaseRequestLine(models.Model):
|
|||||||
else:
|
else:
|
||||||
record.part_number = record.product_id.part_number
|
record.part_number = record.product_id.part_number
|
||||||
record.part_name = record.product_id.part_name
|
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
|
||||||
|
|||||||
47
jikimo_purchase_request/models/stock_picking.py
Normal file
47
jikimo_purchase_request/models/stock_picking.py
Normal file
@@ -0,0 +1,47 @@
|
|||||||
|
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 in purchase_request_lines.mapped('product_id.id')
|
||||||
|
]
|
||||||
|
return res
|
||||||
@@ -1,4 +1,5 @@
|
|||||||
from odoo import api, fields, models
|
from odoo import api, fields, models
|
||||||
|
from collections import defaultdict
|
||||||
|
|
||||||
|
|
||||||
class StockRule(models.Model):
|
class StockRule(models.Model):
|
||||||
@@ -44,13 +45,38 @@ class StockRule(models.Model):
|
|||||||
purchase_request_line_model.create(request_line_data)
|
purchase_request_line_model.create(request_line_data)
|
||||||
|
|
||||||
def _run_buy(self, procurements):
|
def _run_buy(self, procurements):
|
||||||
res = super(StockRule, self)._run_buy(procurements)
|
# 如果补货组相同,并且产品相同,则合并
|
||||||
# 判断是否根据规则生成新的采购申请单据,如果生成则修改状态为 approved
|
procurements_dict = defaultdict()
|
||||||
origins = list(set([procurement[0].origin for procurement in procurements]))
|
for procurement, rule in procurements:
|
||||||
for origin in origins:
|
if (procurement.product_id.id, procurement.values['group_id'], rule.id) not in procurements_dict:
|
||||||
pr_ids = self.env["purchase.request"].sudo().search(
|
procurements_dict[(procurement.product_id.id, procurement.values['group_id'], rule.id)] = {
|
||||||
[('origin', 'like', origin), ('rule_new_add', '=', True), ('state', '=', 'draft')])
|
'product_id': procurement.product_id,
|
||||||
if pr_ids:
|
'product_qty': procurement.product_qty,
|
||||||
pr_ids.write({'need_validation': False})
|
'product_uom': procurement.product_uom,
|
||||||
pr_ids.write({"state": "approved", 'need_validation': True, 'rule_new_add': False})
|
'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
|
return res
|
||||||
|
|||||||
2
jikimo_purchase_request/security/ir.model.access.csv
Normal file
2
jikimo_purchase_request/security/ir.model.access.csv
Normal file
@@ -0,0 +1,2 @@
|
|||||||
|
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
|
||||||
|
3
jikimo_purchase_request/static/src/change.scss
Normal file
3
jikimo_purchase_request/static/src/change.scss
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
th[data-name=keep_description] {
|
||||||
|
min-width: 220px;
|
||||||
|
}
|
||||||
22
jikimo_purchase_request/views/purchase_request_line_view.xml
Normal file
22
jikimo_purchase_request/views/purchase_request_line_view.xml
Normal file
@@ -0,0 +1,22 @@
|
|||||||
|
<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>
|
||||||
@@ -0,0 +1,21 @@
|
|||||||
|
<?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>
|
||||||
|
|
||||||
@@ -15,6 +15,26 @@
|
|||||||
<field name="part_number"/>
|
<field name="part_number"/>
|
||||||
<field name="part_name"/>
|
<field name="part_name"/>
|
||||||
</xpath>
|
</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>
|
</field>
|
||||||
</record>
|
</record>
|
||||||
|
|
||||||
@@ -63,4 +83,9 @@
|
|||||||
</xpath>
|
</xpath>
|
||||||
</field>
|
</field>
|
||||||
</record>
|
</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>
|
</odoo>
|
||||||
21
jikimo_purchase_request/views/stock_picking_views.xml
Normal file
21
jikimo_purchase_request/views/stock_picking_views.xml
Normal file
@@ -0,0 +1,21 @@
|
|||||||
|
<?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,3 +1,4 @@
|
|||||||
# License LGPL-3.0 or later (https://www.gnu.org/licenses/lgpl-3.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_line_make_purchase_order
|
||||||
|
from . import purchase_request_wizard
|
||||||
|
|||||||
@@ -32,6 +32,7 @@ class PurchaseRequestLineMakePurchaseOrder(models.TransientModel):
|
|||||||
line.company_id,
|
line.company_id,
|
||||||
line.request_id.origin,
|
line.request_id.origin,
|
||||||
)
|
)
|
||||||
|
# po_data.update({'related_product':line.related_product.id})
|
||||||
purchase = purchase_obj.create(po_data)
|
purchase = purchase_obj.create(po_data)
|
||||||
|
|
||||||
# Look for any other PO line in the selected PO with same
|
# Look for any other PO line in the selected PO with same
|
||||||
@@ -63,6 +64,8 @@ class PurchaseRequestLineMakePurchaseOrder(models.TransientModel):
|
|||||||
po_line_data = self._prepare_purchase_order_line(purchase, item)
|
po_line_data = self._prepare_purchase_order_line(purchase, item)
|
||||||
if item.keep_description:
|
if item.keep_description:
|
||||||
po_line_data["name"] = item.name
|
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 = po_line_obj.create(po_line_data)
|
||||||
po_line_product_uom_qty = po_line.product_uom._compute_quantity(
|
po_line_product_uom_qty = po_line.product_uom._compute_quantity(
|
||||||
po_line.product_uom_qty, alloc_uom
|
po_line.product_uom_qty, alloc_uom
|
||||||
@@ -101,9 +104,26 @@ class PurchaseRequestLineMakePurchaseOrder(models.TransientModel):
|
|||||||
# 去掉合并必须同一采购组的限制
|
# 去掉合并必须同一采购组的限制
|
||||||
pass
|
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):
|
class PurchaseRequestLineMakePurchaseOrderItem(models.TransientModel):
|
||||||
_inherit = "purchase.request.line.make.purchase.order.item"
|
_inherit = "purchase.request.line.make.purchase.order.item"
|
||||||
|
|
||||||
supply_method = fields.Selection(related='line_id.supply_method', string='供货方式')
|
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,
|
||||||
|
)
|
||||||
|
|||||||
12
jikimo_purchase_request/wizard/purchase_request_wizard.py
Normal file
12
jikimo_purchase_request/wizard/purchase_request_wizard.py
Normal file
@@ -0,0 +1,12 @@
|
|||||||
|
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"})
|
||||||
@@ -0,0 +1,22 @@
|
|||||||
|
<?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,10 +1,9 @@
|
|||||||
# -*- coding: utf-8 -*-
|
# -*- coding: utf-8 -*-
|
||||||
{
|
{
|
||||||
'name': "机企猫 采购审批流程",
|
'name': "机企猫 采购申请审批流程",
|
||||||
|
|
||||||
'summary': """
|
'summary': """
|
||||||
Short (1 phrase/line) summary of the module's purpose, used as
|
采购申请审批流程""",
|
||||||
subtitle on modules listing or apps.openerp.com""",
|
|
||||||
|
|
||||||
'description': """
|
'description': """
|
||||||
Long description of module's purpose
|
Long description of module's purpose
|
||||||
|
|||||||
@@ -1,3 +1,4 @@
|
|||||||
# -*- coding: utf-8 -*-
|
# -*- coding: utf-8 -*-
|
||||||
|
|
||||||
from . import models
|
from . import models
|
||||||
|
from . import stock_rule
|
||||||
@@ -22,3 +22,9 @@ class PurchaseRequest(models.Model):
|
|||||||
self.state = 'approved'
|
self.state = 'approved'
|
||||||
|
|
||||||
return res
|
return res
|
||||||
|
|
||||||
|
@api.model
|
||||||
|
def _get_under_validation_exceptions(self):
|
||||||
|
res = super(PurchaseRequest, self)._get_under_validation_exceptions()
|
||||||
|
res.append("state")
|
||||||
|
return res
|
||||||
|
|||||||
16
jikimo_purchase_request_tier_validation/models/stock_rule.py
Normal file
16
jikimo_purchase_request_tier_validation/models/stock_rule.py
Normal file
@@ -0,0 +1,16 @@
|
|||||||
|
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,12 +1,12 @@
|
|||||||
# -*- coding: utf-8 -*-
|
# -*- coding: utf-8 -*-
|
||||||
{
|
{
|
||||||
'name': "机企猫 采购申请审批流程",
|
'name': "机企猫 采购审批流程",
|
||||||
|
|
||||||
'summary': """
|
'summary': """
|
||||||
采购申请审批流程""",
|
采购审批流程""",
|
||||||
|
|
||||||
'description': """
|
'description': """
|
||||||
采购申请审批流程""",
|
采购审批流程""",
|
||||||
|
|
||||||
'author': "My Company",
|
'author': "My Company",
|
||||||
'website': "https://www.yourcompany.com",
|
'website': "https://www.yourcompany.com",
|
||||||
|
|||||||
2
jikimo_test_generate_product_name/__init__.py
Normal file
2
jikimo_test_generate_product_name/__init__.py
Normal file
@@ -0,0 +1,2 @@
|
|||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
from . import models
|
||||||
18
jikimo_test_generate_product_name/__manifest__.py
Normal file
18
jikimo_test_generate_product_name/__manifest__.py
Normal file
@@ -0,0 +1,18 @@
|
|||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
{
|
||||||
|
'name': 'Jikimo_test_generate_product_name',
|
||||||
|
'version': '',
|
||||||
|
'summary': """ Jikimo_test_generate_product_name Summary """,
|
||||||
|
'author': '',
|
||||||
|
'website': '',
|
||||||
|
'category': '',
|
||||||
|
'depends': ['sf_manufacturing'],
|
||||||
|
'data': [
|
||||||
|
|
||||||
|
],
|
||||||
|
|
||||||
|
'application': True,
|
||||||
|
'installable': True,
|
||||||
|
'auto_install': False,
|
||||||
|
'license': 'LGPL-3',
|
||||||
|
}
|
||||||
2
jikimo_test_generate_product_name/models/__init__.py
Normal file
2
jikimo_test_generate_product_name/models/__init__.py
Normal file
@@ -0,0 +1,2 @@
|
|||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
from . import product_template
|
||||||
21
jikimo_test_generate_product_name/models/product_template.py
Normal file
21
jikimo_test_generate_product_name/models/product_template.py
Normal file
@@ -0,0 +1,21 @@
|
|||||||
|
from odoo import models
|
||||||
|
|
||||||
|
|
||||||
|
class ProductTemplate(models.Model):
|
||||||
|
_inherit = 'product.template'
|
||||||
|
|
||||||
|
def generate_product_name(self, order_id, item, i):
|
||||||
|
"""生成成品名称"""
|
||||||
|
# 3D文件名(去掉后缀,截取前40个字符)+“-”+模型ID
|
||||||
|
product_name = '%s-%s' % ('.'.join(item['model_name'].split('.')[:-1])[:40], item['model_id'])
|
||||||
|
return product_name
|
||||||
|
|
||||||
|
def generate_embryo_name(self, order_id, item, materials_id, materials_type_id, embryo_redundancy_id, i):
|
||||||
|
"""生成坯料名称"""
|
||||||
|
embryo_name = '%s-%s[%s * %s * %s]%s' % (materials_id.name, materials_type_id.name,
|
||||||
|
self.format_float(item['model_long'] + embryo_redundancy_id.long),
|
||||||
|
self.format_float(item['model_width'] + embryo_redundancy_id.width),
|
||||||
|
self.format_float(item['model_height'] + embryo_redundancy_id.height),
|
||||||
|
item['model_id'])
|
||||||
|
return embryo_name
|
||||||
|
|
||||||
@@ -8,7 +8,6 @@
|
|||||||
'category': 'sf',
|
'category': 'sf',
|
||||||
'depends': ['base', 'sf_maintenance', 'jikimo_mini_program'],
|
'depends': ['base', 'sf_maintenance', 'jikimo_mini_program'],
|
||||||
'data': [
|
'data': [
|
||||||
|
|
||||||
],
|
],
|
||||||
|
|
||||||
'application': True,
|
'application': True,
|
||||||
|
|||||||
@@ -1,21 +1,30 @@
|
|||||||
import json
|
import json
|
||||||
from odoo import http
|
from odoo import http
|
||||||
from odoo.http import request
|
from odoo.http import request
|
||||||
from odoo.addons.sf_machine_connect.models.ftp_operate import transfer_nc_files
|
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):
|
class MainController(http.Controller):
|
||||||
|
|
||||||
@http.route('/api/manual_download_program', type='json', methods=['POST'], auth='public', cors='*')
|
@http.route('/api/manual_download_program', type='json', methods=['POST'], auth='wechat_token', cors='*')
|
||||||
|
@api_log('人工线下加工编程文件传输', requester='报工系统')
|
||||||
def manual_download_program(self):
|
def manual_download_program(self):
|
||||||
"""
|
"""
|
||||||
人工线下加工传输编程文件
|
人工线下加工传输编程文件
|
||||||
"""
|
"""
|
||||||
data = json.loads(request.httprequest.data)
|
data = json.loads(request.httprequest.data)
|
||||||
maintenance_equipment_name = data.get('maintenance_equipment_name')
|
maintenance_equipment_id = data.get('maintenance_equipment_id')
|
||||||
model_id = data.get('model_id')
|
model_id = data.get('model_id')
|
||||||
if not maintenance_equipment_name or not model_id:
|
if not maintenance_equipment_id or not model_id:
|
||||||
return {'code': 400, 'message': '参数错误'}
|
return {'code': 400, 'message': '参数错误'}
|
||||||
maintenance_equipment = request.env['maintenance.equipment'].sudo().search([('name', '=', maintenance_equipment_name)], limit=1)
|
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:
|
if not maintenance_equipment:
|
||||||
return {'code': 400, 'message': '机台不存在,请扫描正确的机台二维码'}
|
return {'code': 400, 'message': '机台不存在,请扫描正确的机台二维码'}
|
||||||
product = request.env['product.template'].sudo().search([('model_id', '=', model_id)], limit=1)
|
product = request.env['product.template'].sudo().search([('model_id', '=', model_id)], limit=1)
|
||||||
@@ -45,15 +54,15 @@ class MainController(http.Controller):
|
|||||||
}
|
}
|
||||||
# 传输nc文件
|
# 传输nc文件
|
||||||
try:
|
try:
|
||||||
result = transfer_nc_files(
|
result = transfer_files(
|
||||||
source_ftp_info,
|
source_ftp_info,
|
||||||
target_ftp_info,
|
target_ftp_info,
|
||||||
'/' + str(model_id),
|
'/' + str(model_id),
|
||||||
'/',
|
'/',
|
||||||
match_str=r'^\d*_\d*-' + tool_groups_id.name + r'-\w{2}-all\.nc$'
|
match_str=r'^\d*-' + tool_groups_id.name + r'-\w{2}-all\.nc$'
|
||||||
)
|
)
|
||||||
if result:
|
if len(result) > 0:
|
||||||
return {'code': 200, 'message': 'success'}
|
return {'code': 200, 'message': '传输成功', 'file_list': result}
|
||||||
else:
|
else:
|
||||||
return {'code': 404, 'message': '未找到编程文件'}
|
return {'code': 404, 'message': '未找到编程文件'}
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
|
|||||||
@@ -4,6 +4,7 @@ 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
|
from datetime import datetime
|
||||||
|
|
||||||
_logger = logging.getLogger(__name__)
|
_logger = logging.getLogger(__name__)
|
||||||
@@ -12,6 +13,7 @@ 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):
|
||||||
"""
|
"""
|
||||||
记录工单异常
|
记录工单异常
|
||||||
|
|||||||
@@ -133,6 +133,7 @@ class QualityCheck(models.Model):
|
|||||||
part_name = fields.Char('零件名称', related='product_id.part_name', readonly=False, store=True)
|
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)
|
part_number = fields.Char('零件图号', related='product_id.part_number', readonly=False, store=True)
|
||||||
material_name = fields.Char('材料名称', compute='_compute_material_name')
|
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)
|
# total_qty = fields.Float('总数量', compute='_compute_total_qty', readonly=True)
|
||||||
@@ -141,7 +142,7 @@ class QualityCheck(models.Model):
|
|||||||
# # 出厂检验报告编号
|
# # 出厂检验报告编号
|
||||||
# report_number = fields.Char('出厂检验报告编号', compute='_compute_report_number', readonly=True)
|
# report_number = fields.Char('出厂检验报告编号', compute='_compute_report_number', readonly=True)
|
||||||
# 总数量,值为调拨单_产品明细_数量
|
# 总数量,值为调拨单_产品明细_数量
|
||||||
total_qty = fields.Char('总数量', compute='_compute_total_qty')
|
total_qty = fields.Char('总数量', compute='_compute_total_qty', store=True)
|
||||||
|
|
||||||
column_nums = fields.Integer('测量值列数', default=1)
|
column_nums = fields.Integer('测量值列数', default=1)
|
||||||
|
|
||||||
@@ -153,9 +154,9 @@ class QualityCheck(models.Model):
|
|||||||
for move in record.picking_id.move_ids_without_package:
|
for move in record.picking_id.move_ids_without_package:
|
||||||
if move.product_id == record.product_id:
|
if move.product_id == record.product_id:
|
||||||
total_qty = int(move.product_uom_qty)
|
total_qty = int(move.product_uom_qty)
|
||||||
record.total_qty = total_qty if total_qty > 0 else ''
|
record.total_qty = total_qty if total_qty > 0 else 0
|
||||||
else:
|
else:
|
||||||
record.total_qty = ''
|
record.total_qty = 0
|
||||||
|
|
||||||
# 检验数
|
# 检验数
|
||||||
check_qty = fields.Integer('检验数', default=lambda self: self._get_default_check_qty())
|
check_qty = fields.Integer('检验数', default=lambda self: self._get_default_check_qty())
|
||||||
@@ -338,7 +339,7 @@ class QualityCheck(models.Model):
|
|||||||
|
|
||||||
# 4. 获取报告动作并生成PDF(此时二维码将包含正确的文档ID)
|
# 4. 获取报告动作并生成PDF(此时二维码将包含正确的文档ID)
|
||||||
report_action = self.env.ref('sf_quality.action_report_quality_inspection')
|
report_action = self.env.ref('sf_quality.action_report_quality_inspection')
|
||||||
pdf_content, _ = report_action._render_qweb_pdf(
|
pdf_content, v = report_action._render_qweb_pdf(
|
||||||
report_ref=report_action.report_name,
|
report_ref=report_action.report_name,
|
||||||
res_ids=self.ids
|
res_ids=self.ids
|
||||||
)
|
)
|
||||||
@@ -735,8 +736,9 @@ 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,
|
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")
|
precision_rounding=rounding, rounding_method="UP")
|
||||||
else:
|
else:
|
||||||
qc.qty_to_test = qc.qty_line
|
qc.qty_to_test = qc.qty_line
|
||||||
|
|
||||||
|
|||||||
@@ -493,6 +493,9 @@
|
|||||||
<field name="picking_id"/>
|
<field name="picking_id"/>
|
||||||
<field name="lot_id"/>
|
<field name="lot_id"/>
|
||||||
<field name="team_id"/>
|
<field name="team_id"/>
|
||||||
|
<field name="part_number"/>
|
||||||
|
<field name="part_name"/>
|
||||||
|
<field name="model_id"/>
|
||||||
<filter string="In Progress" name="progress" domain="[('quality_state', '=', 'none')]"/>
|
<filter string="In Progress" name="progress" domain="[('quality_state', '=', 'none')]"/>
|
||||||
<filter string="Passed" name="passed" domain="[('quality_state', '=', 'pass')]"/>
|
<filter string="Passed" name="passed" domain="[('quality_state', '=', 'pass')]"/>
|
||||||
<filter string="Failed" name="failed" domain="[('quality_state', '=', 'fail')]"/>
|
<filter string="Failed" name="failed" domain="[('quality_state', '=', 'fail')]"/>
|
||||||
|
|||||||
@@ -23,8 +23,8 @@ class QualityCheckWizard(models.TransientModel):
|
|||||||
lot_name = fields.Char(related='current_check_id.lot_name')
|
lot_name = fields.Char(related='current_check_id.lot_name')
|
||||||
lot_line_id = fields.Many2one(related='current_check_id.lot_line_id')
|
lot_line_id = fields.Many2one(related='current_check_id.lot_line_id')
|
||||||
qty_line = fields.Float(related='current_check_id.qty_line')
|
qty_line = fields.Float(related='current_check_id.qty_line')
|
||||||
qty_to_test = fields.Float(related='current_check_id.qty_to_test')
|
qty_to_test = fields.Float(related='current_check_id.qty_to_test', string='待检')
|
||||||
qty_tested = fields.Float(related='current_check_id.qty_tested', readonly=False)
|
qty_tested = fields.Float(related='current_check_id.qty_tested', string='已检', readonly=False)
|
||||||
measure = fields.Float(related='current_check_id.measure', readonly=False)
|
measure = fields.Float(related='current_check_id.measure', readonly=False)
|
||||||
measure_on = fields.Selection(related='current_check_id.measure_on')
|
measure_on = fields.Selection(related='current_check_id.measure_on')
|
||||||
quality_state = fields.Selection(related='current_check_id.quality_state')
|
quality_state = fields.Selection(related='current_check_id.quality_state')
|
||||||
|
|||||||
@@ -1,3 +1,7 @@
|
|||||||
pystrich
|
pystrich
|
||||||
cpca
|
cpca==0.5.5
|
||||||
pycryptodome==3.20
|
wechatpy==1.8.18
|
||||||
|
pycryptodome==3.22.0
|
||||||
|
openupgradelib==3.10.0
|
||||||
|
opcua==0.98.13
|
||||||
|
openpyxl
|
||||||
@@ -1,3 +1,4 @@
|
|||||||
from . import models
|
from . import models
|
||||||
from . import commons
|
from . import commons
|
||||||
from . import controllers
|
from . import controllers
|
||||||
|
from . import decorators
|
||||||
|
|||||||
@@ -25,7 +25,7 @@
|
|||||||
'views/menu_fixture_view.xml',
|
'views/menu_fixture_view.xml',
|
||||||
'views/change_base_view.xml',
|
'views/change_base_view.xml',
|
||||||
'views/Printer.xml',
|
'views/Printer.xml',
|
||||||
|
'views/api_log_views.xml',
|
||||||
],
|
],
|
||||||
'demo': [
|
'demo': [
|
||||||
],
|
],
|
||||||
|
|||||||
@@ -103,12 +103,19 @@ class PrintingUtils(models.AbstractModel):
|
|||||||
self.send_to_printer(host, port, zpl_code)
|
self.send_to_printer(host, port, zpl_code)
|
||||||
|
|
||||||
|
|
||||||
def add_qr_code_to_pdf(self, pdf_path:str, content:str, buttom_text:Optional[str]=False):
|
def add_qr_code_to_pdf(
|
||||||
|
self,
|
||||||
|
pdf_path:str,
|
||||||
|
content:str,
|
||||||
|
qr_code_buttom_text:Optional[str]=False,
|
||||||
|
buttom_text:Optional[str]=False,
|
||||||
|
):
|
||||||
"""
|
"""
|
||||||
在PDF文件中添加二维码
|
在PDF文件中添加二维码
|
||||||
:param pdf_path: PDF文件路径
|
:param pdf_path: PDF文件路径
|
||||||
:param content: 二维码内容
|
:param content: 二维码内容
|
||||||
:param buttom_text: 二维码下方文字
|
:param qr_code_buttom_text: 二维码下方文字
|
||||||
|
:param buttom_text: 正文下方文字
|
||||||
:return: 是否成功
|
:return: 是否成功
|
||||||
"""
|
"""
|
||||||
if not os.path.exists(pdf_path):
|
if not os.path.exists(pdf_path):
|
||||||
@@ -134,7 +141,7 @@ class PrintingUtils(models.AbstractModel):
|
|||||||
|
|
||||||
# 注册中文字体
|
# 注册中文字体
|
||||||
font_paths = [
|
font_paths = [
|
||||||
"/usr/share/fonts/windows/simsun.ttc", # Windows系统宋体
|
"/usr/share/fonts/chinese/simsun.ttc", # Windows系统宋体
|
||||||
"c:/windows/fonts/simsun.ttc", # Windows系统宋体另一个位置
|
"c:/windows/fonts/simsun.ttc", # Windows系统宋体另一个位置
|
||||||
"/usr/share/fonts/truetype/droid/DroidSansFallbackFull.ttf", # Linux Droid字体
|
"/usr/share/fonts/truetype/droid/DroidSansFallbackFull.ttf", # Linux Droid字体
|
||||||
"/usr/share/fonts/truetype/wqy/wqy-zenhei.ttc", # 文泉驿正黑
|
"/usr/share/fonts/truetype/wqy/wqy-zenhei.ttc", # 文泉驿正黑
|
||||||
@@ -156,8 +163,9 @@ class PrintingUtils(models.AbstractModel):
|
|||||||
existing_pdf = PdfFileReader(original_file)
|
existing_pdf = PdfFileReader(original_file)
|
||||||
output = PdfFileWriter()
|
output = PdfFileWriter()
|
||||||
|
|
||||||
# 处理第一页
|
# 处理最后一页
|
||||||
page = existing_pdf.getPage(0)
|
last_page = existing_pdf.getNumPages() - 1
|
||||||
|
page = existing_pdf.getPage(last_page)
|
||||||
# 获取页面尺寸
|
# 获取页面尺寸
|
||||||
page_width = float(page.mediaBox.getWidth())
|
page_width = float(page.mediaBox.getWidth())
|
||||||
page_height = float(page.mediaBox.getHeight())
|
page_height = float(page.mediaBox.getHeight())
|
||||||
@@ -167,10 +175,10 @@ class PrintingUtils(models.AbstractModel):
|
|||||||
|
|
||||||
# 设置字体
|
# 设置字体
|
||||||
if font_found:
|
if font_found:
|
||||||
c.setFont('SimSun', 14) # 增大字体大小到14pt
|
c.setFont('SimSun', 10) # 增大字体大小到14pt
|
||||||
else:
|
else:
|
||||||
# 如果没有找到中文字体,使用默认字体
|
# 如果没有找到中文字体,使用默认字体
|
||||||
c.setFont('Helvetica', 14)
|
c.setFont('Helvetica', 10)
|
||||||
logging.warning("未找到中文字体,将使用默认字体")
|
logging.warning("未找到中文字体,将使用默认字体")
|
||||||
|
|
||||||
# 在右下角绘制二维码,预留边距
|
# 在右下角绘制二维码,预留边距
|
||||||
@@ -179,13 +187,29 @@ class PrintingUtils(models.AbstractModel):
|
|||||||
qr_y = margin + 20 # 将二维码向上移动一点,为文字留出空间
|
qr_y = margin + 20 # 将二维码向上移动一点,为文字留出空间
|
||||||
c.drawImage(qr_temp_path, page_width - qr_size - margin, qr_y, width=qr_size, height=qr_size)
|
c.drawImage(qr_temp_path, page_width - qr_size - margin, qr_y, width=qr_size, height=qr_size)
|
||||||
|
|
||||||
if buttom_text:
|
if qr_code_buttom_text:
|
||||||
# 在二维码下方绘制文字
|
# 在二维码下方绘制文字
|
||||||
text = buttom_text
|
text = qr_code_buttom_text
|
||||||
text_width = c.stringWidth(text, "SimSun" if font_found else "Helvetica", 14) # 准确计算文字宽度
|
text_width = c.stringWidth(text, "SimSun" if font_found else "Helvetica", 10) # 准确计算文字宽度
|
||||||
text_x = page_width - qr_size - margin + (qr_size - text_width) / 2 # 文字居中对齐
|
text_x = page_width - qr_size - margin + (qr_size - text_width) / 2 # 文字居中对齐
|
||||||
text_y = margin + 20 # 文字位置靠近底部
|
text_y = margin + 20 # 文字位置靠近底部
|
||||||
c.drawString(text_x, text_y, text)
|
c.drawString(text_x, text_y, text)
|
||||||
|
|
||||||
|
# 设置字体
|
||||||
|
if font_found:
|
||||||
|
c.setFont('SimSun', 12) # 增大字体大小到14pt
|
||||||
|
else:
|
||||||
|
# 如果没有找到中文字体,使用默认字体
|
||||||
|
c.setFont('Helvetica', 120)
|
||||||
|
logging.warning("未找到中文字体,将使用默认字体")
|
||||||
|
|
||||||
|
if buttom_text:
|
||||||
|
# 在下方中间添加文字
|
||||||
|
text = buttom_text
|
||||||
|
text_width = c.stringWidth(text, "SimSun" if font_found else "Helvetica", 12) # 准确计算文字宽度
|
||||||
|
text_x = (page_width - text_width) / 2 # 文字居中对齐
|
||||||
|
text_y = margin + 20 # 文字位置靠近底部
|
||||||
|
c.drawString(text_x, text_y, text)
|
||||||
|
|
||||||
c.save()
|
c.save()
|
||||||
|
|
||||||
@@ -196,11 +220,12 @@ class PrintingUtils(models.AbstractModel):
|
|||||||
|
|
||||||
# 合并原始页面和二维码页面
|
# 合并原始页面和二维码页面
|
||||||
page.mergePage(qr_page)
|
page.mergePage(qr_page)
|
||||||
output.addPage(page)
|
|
||||||
|
|
||||||
# 添加剩余的页面
|
# 添加剩余的页面
|
||||||
for i in range(1, existing_pdf.getNumPages()):
|
for i in range(0, last_page):
|
||||||
output.addPage(existing_pdf.getPage(i))
|
output.addPage(existing_pdf.getPage(i))
|
||||||
|
|
||||||
|
output.addPage(page)
|
||||||
|
|
||||||
# 保存最终的PDF到一个临时文件
|
# 保存最终的PDF到一个临时文件
|
||||||
final_temp_path = pdf_path + '.tmp'
|
final_temp_path = pdf_path + '.tmp'
|
||||||
|
|||||||
@@ -1,15 +1,18 @@
|
|||||||
# -*- coding: utf-8 -*-
|
# -*- coding: utf-8 -*-
|
||||||
import logging
|
import logging
|
||||||
import json
|
import json
|
||||||
import base64
|
import logging
|
||||||
from odoo import http
|
from odoo import http
|
||||||
from odoo.http import request
|
from odoo.http import request
|
||||||
|
from odoo.addons.sf_base.decorators.api_log import api_log
|
||||||
|
|
||||||
|
_logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
class Manufacturing_Connect(http.Controller):
|
class Manufacturing_Connect(http.Controller):
|
||||||
|
|
||||||
@http.route('/AutoDeviceApi/MachineToolGroup', type='json', auth='sf_token', methods=['GET', 'POST'], csrf=False,
|
@http.route('/AutoDeviceApi/MachineToolGroup', type='json', auth='sf_token', methods=['GET', 'POST'], csrf=False,
|
||||||
cors="*")
|
cors="*")
|
||||||
|
@api_log('机床刀具组', requester='中控系统')
|
||||||
def get_maintenance_tool_groups_Info(self, **kw):
|
def get_maintenance_tool_groups_Info(self, **kw):
|
||||||
"""
|
"""
|
||||||
机床刀具组接口
|
机床刀具组接口
|
||||||
|
|||||||
1
sf_base/decorators/__init__.py
Normal file
1
sf_base/decorators/__init__.py
Normal file
@@ -0,0 +1 @@
|
|||||||
|
from . import api_log
|
||||||
65
sf_base/decorators/api_log.py
Normal file
65
sf_base/decorators/api_log.py
Normal file
@@ -0,0 +1,65 @@
|
|||||||
|
|
||||||
|
import functools
|
||||||
|
import json
|
||||||
|
import logging
|
||||||
|
from datetime import datetime
|
||||||
|
from odoo.http import request
|
||||||
|
|
||||||
|
_logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
def api_log(name=None, requester=None):
|
||||||
|
"""记录API请求日志的装饰器"""
|
||||||
|
def decorator(func):
|
||||||
|
@functools.wraps(func)
|
||||||
|
def wrapper(*args, **kwargs):
|
||||||
|
start_time = datetime.now()
|
||||||
|
|
||||||
|
# 获取请求信息
|
||||||
|
try:
|
||||||
|
# 获取请求数据
|
||||||
|
request_data = json.loads(request.httprequest.data) if request.httprequest.data else {}
|
||||||
|
# 获取请求路径
|
||||||
|
path = request.httprequest.path
|
||||||
|
# 获取请求方法
|
||||||
|
method = request.httprequest.method
|
||||||
|
# 获取客户端IP
|
||||||
|
remote_addr = request.httprequest.remote_addr
|
||||||
|
|
||||||
|
# 执行原始函数
|
||||||
|
result = func(*args, **kwargs)
|
||||||
|
origin_result = result
|
||||||
|
if isinstance(result, str):
|
||||||
|
result = json.loads(result)
|
||||||
|
|
||||||
|
# 计算响应时间
|
||||||
|
end_time = datetime.now()
|
||||||
|
response_time = (end_time - start_time).total_seconds()
|
||||||
|
|
||||||
|
# 获取响应状态
|
||||||
|
status = result.get('code') if 'code' in result else result.get('ErrorCode') if 'ErrorCode' in result else 200
|
||||||
|
|
||||||
|
# 创建日志记录
|
||||||
|
log_vals = {
|
||||||
|
'name': name or func.__name__,
|
||||||
|
'path': path,
|
||||||
|
'method': method.upper(),
|
||||||
|
'request_data': json.dumps(request_data, ensure_ascii=False),
|
||||||
|
'response_data': json.dumps(result, ensure_ascii=False),
|
||||||
|
'remote_addr': remote_addr,
|
||||||
|
'response_time': response_time,
|
||||||
|
'status': 200 if status == 0 else status,
|
||||||
|
'requester': requester,
|
||||||
|
'responser': '智能工厂'
|
||||||
|
}
|
||||||
|
|
||||||
|
# 异步创建日志记录
|
||||||
|
request.env['api.request.log'].sudo().with_context(tracking_disable=True).create(log_vals)
|
||||||
|
|
||||||
|
return origin_result
|
||||||
|
|
||||||
|
except Exception as e:
|
||||||
|
_logger.error(f"API日志记录失败: {str(e)}")
|
||||||
|
# 即使日志记录失败,也要返回原始结果
|
||||||
|
return func(*args, **kwargs)
|
||||||
|
return wrapper
|
||||||
|
return decorator
|
||||||
@@ -6,3 +6,4 @@ from . import functional_fixture
|
|||||||
from . import tool_other_features
|
from . import tool_other_features
|
||||||
from . import basic_parameters_fixture
|
from . import basic_parameters_fixture
|
||||||
from . import ir_sequence
|
from . import ir_sequence
|
||||||
|
from . import api_log
|
||||||
|
|||||||
72
sf_base/models/api_log.py
Normal file
72
sf_base/models/api_log.py
Normal file
@@ -0,0 +1,72 @@
|
|||||||
|
from odoo import models, fields, api
|
||||||
|
import json, ast
|
||||||
|
import logging
|
||||||
|
import requests
|
||||||
|
|
||||||
|
_logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
|
class ApiRequestLog(models.Model):
|
||||||
|
_name = 'api.request.log'
|
||||||
|
_description = '接口请求日志'
|
||||||
|
_order = 'id desc'
|
||||||
|
|
||||||
|
name = fields.Char('接口名称')
|
||||||
|
path = fields.Char('请求路径')
|
||||||
|
method = fields.Char('请求方法')
|
||||||
|
request_data = fields.Text('请求数据')
|
||||||
|
response_data = fields.Text('响应数据')
|
||||||
|
remote_addr = fields.Char('客户端IP')
|
||||||
|
response_time = fields.Float('响应时间(秒)', digits=(16, 6))
|
||||||
|
status = fields.Integer('状态码')
|
||||||
|
requester = fields.Char('请求方')
|
||||||
|
responser = fields.Char('响应方')
|
||||||
|
|
||||||
|
@api.model
|
||||||
|
def log_request(self, method, url, name=None, responser=None, **kwargs):
|
||||||
|
# Log the request
|
||||||
|
request_headers = kwargs.get('headers', {})
|
||||||
|
request_body = kwargs.get('json') or kwargs.get('params') or {}
|
||||||
|
|
||||||
|
_logger.info(f"Request: {method} {url} Headers: {request_headers} Body: {request_body}")
|
||||||
|
|
||||||
|
# Make the actual request
|
||||||
|
response = requests.request(method, url, **kwargs)
|
||||||
|
|
||||||
|
# Log the response
|
||||||
|
response_status = response.status_code
|
||||||
|
response_headers = response.headers
|
||||||
|
response_body = response.text
|
||||||
|
response_time = response.elapsed.total_seconds()
|
||||||
|
|
||||||
|
_logger.info(f"Response: Status: {response_status} Headers: {response_headers} Body: {response_body}")
|
||||||
|
|
||||||
|
try:
|
||||||
|
# 如果是字符串,先尝试用 ast.literal_eval 安全地转换成 Python 对象
|
||||||
|
if isinstance(response_body, str):
|
||||||
|
|
||||||
|
response_body_obj = json.loads(response_body)
|
||||||
|
else:
|
||||||
|
response_body_obj = response_body
|
||||||
|
|
||||||
|
# 再使用 json.dumps 转换成标准的 JSON 字符串
|
||||||
|
response_body = json.dumps(response_body_obj, ensure_ascii=False)
|
||||||
|
except Exception as e:
|
||||||
|
_logger.warning(f"转换 response_body 到标准 JSON 失败: {str(e)}")
|
||||||
|
# 如果转换失败,保持原样
|
||||||
|
|
||||||
|
# Save to database
|
||||||
|
self.sudo().create({
|
||||||
|
'name': name,
|
||||||
|
'path': url,
|
||||||
|
'method': method.upper(),
|
||||||
|
'request_data': request_body,
|
||||||
|
'response_data': response_body,
|
||||||
|
'remote_addr': None,
|
||||||
|
'response_time': response_time,
|
||||||
|
'status': response_status,
|
||||||
|
'requester': '智能工厂',
|
||||||
|
'responser': responser
|
||||||
|
})
|
||||||
|
|
||||||
|
return response
|
||||||
@@ -254,3 +254,6 @@ access_sf_machining_accuracy_admin,sf_machining_accuracy_admin,model_sf_machinin
|
|||||||
|
|
||||||
access_sf_embryo_redundancy,sf_embryo_redundancy,model_sf_embryo_redundancy,base.group_user,1,0,0,0
|
access_sf_embryo_redundancy,sf_embryo_redundancy,model_sf_embryo_redundancy,base.group_user,1,0,0,0
|
||||||
access_sf_embryo_redundancy_admin,sf_embryo_redundancy_admin,model_sf_embryo_redundancy,base.group_system,1,0,0,0
|
access_sf_embryo_redundancy_admin,sf_embryo_redundancy_admin,model_sf_embryo_redundancy,base.group_system,1,0,0,0
|
||||||
|
|
||||||
|
access_api_request_log_user,api.request.log.user,model_api_request_log,base.group_user,1,0,0,0
|
||||||
|
access_api_request_log_admin,api.request.log.admin,model_api_request_log,base.group_system,1,1,1,1
|
||||||
|
83
sf_base/views/api_log_views.xml
Normal file
83
sf_base/views/api_log_views.xml
Normal file
@@ -0,0 +1,83 @@
|
|||||||
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
|
<odoo>
|
||||||
|
<record id="view_api_request_log_tree" model="ir.ui.view">
|
||||||
|
<field name="name">api.request.log.tree</field>
|
||||||
|
<field name="model">api.request.log</field>
|
||||||
|
<field name="arch" type="xml">
|
||||||
|
<tree>
|
||||||
|
<field name="name"/>
|
||||||
|
<field name="path"/>
|
||||||
|
<field name="method"/>
|
||||||
|
<field name="remote_addr"/>
|
||||||
|
<field name="response_time" sum="0"/>
|
||||||
|
<field name="requester"/>
|
||||||
|
<field name="responser"/>
|
||||||
|
<field name="create_date" string="请求时间"/>
|
||||||
|
<field name="status" sum="0"/>
|
||||||
|
</tree>
|
||||||
|
</field>
|
||||||
|
</record>
|
||||||
|
|
||||||
|
<record id="view_api_request_log_form" model="ir.ui.view">
|
||||||
|
<field name="name">api.request.log.form</field>
|
||||||
|
<field name="model">api.request.log</field>
|
||||||
|
<field name="arch" type="xml">
|
||||||
|
<form>
|
||||||
|
<sheet>
|
||||||
|
<group>
|
||||||
|
<group>
|
||||||
|
<field name="name"/>
|
||||||
|
<field name="path"/>
|
||||||
|
<field name="method"/>
|
||||||
|
<field name="remote_addr"/>
|
||||||
|
</group>
|
||||||
|
<group>
|
||||||
|
<field name="response_time"/>
|
||||||
|
<field name="status"/>
|
||||||
|
<field name="requester"/>
|
||||||
|
<field name="responser"/>
|
||||||
|
<field name="create_date" string="请求时间"/>
|
||||||
|
</group>
|
||||||
|
</group>
|
||||||
|
<notebook>
|
||||||
|
<page string="请求数据">
|
||||||
|
<field name="request_data"/>
|
||||||
|
</page>
|
||||||
|
<page string="响应数据">
|
||||||
|
<field name="response_data"/>
|
||||||
|
</page>
|
||||||
|
</notebook>
|
||||||
|
</sheet>
|
||||||
|
</form>
|
||||||
|
</field>
|
||||||
|
</record>
|
||||||
|
|
||||||
|
<record model="ir.ui.view" id="view_api_request_log_search">
|
||||||
|
<field name="name">api.request.log.search</field>
|
||||||
|
<field name="model">api.request.log</field>
|
||||||
|
<field name="arch" type="xml">
|
||||||
|
<search string="API请求日志">
|
||||||
|
<field name="name"/>
|
||||||
|
<field name="requester"/>
|
||||||
|
<field name="responser"/>
|
||||||
|
<group>
|
||||||
|
<filter name="name" context="{'group_by':'name'}"/>
|
||||||
|
<filter name="requester" context="{'group_by':'requester'}"/>
|
||||||
|
<filter name="responser" context="{'group_by':'responser'}"/>
|
||||||
|
</group>
|
||||||
|
</search>
|
||||||
|
</field>
|
||||||
|
</record>
|
||||||
|
|
||||||
|
<record id="action_api_request_log" model="ir.actions.act_window">
|
||||||
|
<field name="name">API请求日志</field>
|
||||||
|
<field name="res_model">api.request.log</field>
|
||||||
|
<field name="view_mode">tree,form</field>
|
||||||
|
</record>
|
||||||
|
|
||||||
|
<menuitem id="menu_api_request_log"
|
||||||
|
name="API请求日志"
|
||||||
|
parent="base.next_id_9"
|
||||||
|
action="action_api_request_log"
|
||||||
|
sequence="100"/>
|
||||||
|
</odoo>
|
||||||
@@ -2,7 +2,7 @@
|
|||||||
import logging
|
import logging
|
||||||
from datetime import datetime, timedelta
|
from datetime import datetime, timedelta
|
||||||
import hashlib
|
import hashlib
|
||||||
from odoo import models
|
from odoo import models, SUPERUSER_ID
|
||||||
from odoo.http import request
|
from odoo.http import request
|
||||||
|
|
||||||
__author__ = 'jinling.yang'
|
__author__ = 'jinling.yang'
|
||||||
@@ -48,5 +48,7 @@ class Http(models.AbstractModel):
|
|||||||
_logger.info('sf_secret_key:%s' % factory_secret.sf_secret_key)
|
_logger.info('sf_secret_key:%s' % factory_secret.sf_secret_key)
|
||||||
if check_sf_str != datas['HTTP_CHECKSTR']:
|
if check_sf_str != datas['HTTP_CHECKSTR']:
|
||||||
raise AuthenticationError('数据校验不通过')
|
raise AuthenticationError('数据校验不通过')
|
||||||
|
# 设置管理员用户
|
||||||
|
request.update_env(user=SUPERUSER_ID)
|
||||||
else:
|
else:
|
||||||
raise AuthenticationError('请求参数中无token')
|
raise AuthenticationError('请求参数中无token')
|
||||||
|
|||||||
3
sf_demand_plan/__init__.py
Normal file
3
sf_demand_plan/__init__.py
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
from . import models
|
||||||
|
from . import wizard
|
||||||
33
sf_demand_plan/__manifest__.py
Normal file
33
sf_demand_plan/__manifest__.py
Normal file
@@ -0,0 +1,33 @@
|
|||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
# Part of Odoo. See LICENSE file for full copyright and licensing details.
|
||||||
|
{
|
||||||
|
'name': '机企猫智能工厂 需求计划',
|
||||||
|
'version': '1.0',
|
||||||
|
'summary': '智能工厂计划管理',
|
||||||
|
'sequence': 1,
|
||||||
|
'description': """
|
||||||
|
在本模块,支持齐套检查与下达生产
|
||||||
|
""",
|
||||||
|
'category': 'sf',
|
||||||
|
'website': 'https://www.sf.jikimo.com',
|
||||||
|
'depends': ['sf_plan', 'jikimo_printing'],
|
||||||
|
'data': [
|
||||||
|
'security/ir.model.access.csv',
|
||||||
|
'views/demand_plan.xml',
|
||||||
|
'wizard/sf_demand_plan_print_wizard_view.xml',
|
||||||
|
],
|
||||||
|
'demo': [
|
||||||
|
],
|
||||||
|
'assets': {
|
||||||
|
'web.assets_qweb': [
|
||||||
|
],
|
||||||
|
'web.assets_backend': [
|
||||||
|
'sf_demand_plan/static/src/scss/style.css',
|
||||||
|
'sf_demand_plan/static/src/js/print_demand.js',
|
||||||
|
]
|
||||||
|
},
|
||||||
|
'license': 'LGPL-3',
|
||||||
|
'installable': True,
|
||||||
|
'application': False,
|
||||||
|
'auto_install': False,
|
||||||
|
}
|
||||||
4
sf_demand_plan/models/__init__.py
Normal file
4
sf_demand_plan/models/__init__.py
Normal file
@@ -0,0 +1,4 @@
|
|||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
|
||||||
|
from . import sf_production_demand_plan
|
||||||
|
from . import sale_order
|
||||||
29
sf_demand_plan/models/sale_order.py
Normal file
29
sf_demand_plan/models/sale_order.py
Normal file
@@ -0,0 +1,29 @@
|
|||||||
|
from odoo import models, fields, api, _
|
||||||
|
|
||||||
|
|
||||||
|
class ReSaleOrder(models.Model):
|
||||||
|
_inherit = 'sale.order'
|
||||||
|
|
||||||
|
mrp_production_ids = fields.Many2many(
|
||||||
|
'mrp.production',
|
||||||
|
compute='_compute_mrp_production_ids',
|
||||||
|
string='与此销售订单相关联的制造订单',
|
||||||
|
groups='mrp.group_mrp_user', store=True)
|
||||||
|
|
||||||
|
def sale_order_create_line(self, product, item):
|
||||||
|
ret = super(ReSaleOrder, self).sale_order_create_line(product, item)
|
||||||
|
vals = {
|
||||||
|
'sale_order_id': ret.order_id.id,
|
||||||
|
'sale_order_line_id': ret.id,
|
||||||
|
}
|
||||||
|
demand_plan = self.env['sf.production.demand.plan'].sudo().create(vals)
|
||||||
|
if demand_plan.product_id.machining_drawings_name:
|
||||||
|
filename_url = demand_plan.product_id.machining_drawings_name.rsplit('.', 1)[0]
|
||||||
|
wizard_vals = {
|
||||||
|
'demand_plan_id': demand_plan.id,
|
||||||
|
'model_id': demand_plan.model_id,
|
||||||
|
'filename_url': filename_url,
|
||||||
|
'type': '1',
|
||||||
|
}
|
||||||
|
self.env['sf.demand.plan.print.wizard'].sudo().create(wizard_vals)
|
||||||
|
return ret
|
||||||
550
sf_demand_plan/models/sf_production_demand_plan.py
Normal file
550
sf_demand_plan/models/sf_production_demand_plan.py
Normal file
@@ -0,0 +1,550 @@
|
|||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
import ast
|
||||||
|
import json
|
||||||
|
from odoo import models, fields, api, _
|
||||||
|
from odoo.exceptions import ValidationError
|
||||||
|
from odoo.tools import float_compare
|
||||||
|
from datetime import datetime, timedelta
|
||||||
|
from odoo.exceptions import UserError
|
||||||
|
|
||||||
|
|
||||||
|
class SfProductionDemandPlan(models.Model):
|
||||||
|
_name = 'sf.production.demand.plan'
|
||||||
|
_description = 'sf_production_demand_plan'
|
||||||
|
|
||||||
|
def _get_machining_precision(self):
|
||||||
|
machinings = self.env['sf.machining.accuracy'].sudo().search([])
|
||||||
|
list = [(m.sync_id, m.name) for m in machinings]
|
||||||
|
return list
|
||||||
|
|
||||||
|
priority = fields.Selection([
|
||||||
|
('1', '紧急'),
|
||||||
|
('2', '高'),
|
||||||
|
('3', '中'),
|
||||||
|
('4', '低'),
|
||||||
|
], string='优先级', default='3')
|
||||||
|
status = fields.Selection([
|
||||||
|
('10', '草稿'),
|
||||||
|
('20', '待确认'),
|
||||||
|
('30', '需求确认'),
|
||||||
|
('50', '待下达生产'),
|
||||||
|
('60', '已下达'),
|
||||||
|
('100', '取消'),
|
||||||
|
], string='状态', compute='_compute_status', store=True)
|
||||||
|
sale_order_id = fields.Many2one(comodel_name="sale.order",
|
||||||
|
string="销售订单", readonly=True)
|
||||||
|
sale_order_line_id = fields.Many2one(comodel_name="sale.order.line",
|
||||||
|
string="销售订单明细", readonly=True)
|
||||||
|
sale_order_line_number = fields.Char(string='销售订单行', compute='_compute_sale_order_line_number', store=True)
|
||||||
|
company_id = fields.Many2one(
|
||||||
|
related='sale_order_id.company_id',
|
||||||
|
store=True, index=True, precompute=True)
|
||||||
|
customer_name = fields.Char('客户', related='sale_order_id.customer_name')
|
||||||
|
order_remark = fields.Text(related='sale_order_id.remark',
|
||||||
|
string="订单备注", store=True)
|
||||||
|
glb_url = fields.Char(related='sale_order_line_id.glb_url', string='glb文件地址')
|
||||||
|
product_id = fields.Many2one(
|
||||||
|
comodel_name='product.product',
|
||||||
|
related='sale_order_line_id.product_id',
|
||||||
|
string='产品', store=True, index=True)
|
||||||
|
model_id = fields.Char('模型ID', related='product_id.model_id')
|
||||||
|
part_name = fields.Char('零件名称', related='product_id.part_name')
|
||||||
|
part_number = fields.Char('零件图号', compute='_compute_part_number', store=True)
|
||||||
|
is_incoming_material = fields.Boolean('客供料', related='sale_order_line_id.is_incoming_material', store=True)
|
||||||
|
supply_method = fields.Selection([
|
||||||
|
('automation', "自动化产线加工"),
|
||||||
|
('manual', "人工线下加工"),
|
||||||
|
('purchase', "外购"),
|
||||||
|
('outsourcing', "委外加工"),
|
||||||
|
], string='供货方式', related='sale_order_line_id.supply_method', store=True)
|
||||||
|
product_uom_qty = fields.Float(
|
||||||
|
string="需求数量",
|
||||||
|
related='sale_order_line_id.product_uom_qty', store=True)
|
||||||
|
deadline_of_delivery = fields.Date('客户交期', related='sale_order_line_id.delivery_end_date', store=True)
|
||||||
|
inventory_quantity_auto_apply = fields.Float(
|
||||||
|
string="成品库存",
|
||||||
|
compute='_compute_inventory_quantity_auto_apply'
|
||||||
|
)
|
||||||
|
qty_delivered = fields.Float(
|
||||||
|
"交货数量", related='sale_order_line_id.qty_delivered')
|
||||||
|
qty_to_deliver = fields.Float(
|
||||||
|
"待交货数量", related='sale_order_line_id.qty_to_deliver')
|
||||||
|
model_long = fields.Char('尺寸(mm)', compute='_compute_model_long')
|
||||||
|
blank_type = fields.Selection([('圆料', '圆料'), ('方料', '方料')], string='坯料分类',
|
||||||
|
related='product_id.blank_type')
|
||||||
|
blank_precision = fields.Selection([('精坯', '精坯'), ('粗坯', '粗坯')], string='坯料类型', related='product_id.blank_precision')
|
||||||
|
embryo_long = fields.Char('坯料尺寸(mm)', compute='_compute_embryo_long')
|
||||||
|
materials_id = fields.Char('材料', compute='_compute_materials_id', store=True)
|
||||||
|
model_machining_precision = fields.Selection(selection=_get_machining_precision, string='精度',
|
||||||
|
related='product_id.model_machining_precision')
|
||||||
|
model_process_parameters_ids = fields.Many2many('sf.production.process.parameter',
|
||||||
|
'plan_process_parameter_rel',
|
||||||
|
string='表面工艺',
|
||||||
|
compute='_compute_model_process_parameters_ids'
|
||||||
|
, store=True
|
||||||
|
)
|
||||||
|
product_remark = fields.Char("产品备注", related='product_id.model_remark')
|
||||||
|
order_code = fields.Char('E-SHOP订单号', related='sale_order_id.order_code')
|
||||||
|
order_state = fields.Selection(
|
||||||
|
string='订单状态',
|
||||||
|
related='sale_order_line_id.state')
|
||||||
|
route_id = fields.Many2one('stock.route', string='路线', related='sale_order_line_id.route_id', store=True)
|
||||||
|
contract_date = fields.Date('合同日期', related='sale_order_id.contract_date')
|
||||||
|
date_order = fields.Datetime('下单日期', related='sale_order_id.date_order')
|
||||||
|
contract_code = fields.Char('合同号', related='sale_order_id.contract_code', store=True)
|
||||||
|
plan_remark = fields.Text("计划备注")
|
||||||
|
material_check = fields.Selection([
|
||||||
|
('0', "未齐套"),
|
||||||
|
('1', "已齐套"),
|
||||||
|
], string='投料齐套检查', compute='_compute_material_check', store=True)
|
||||||
|
processing_time = fields.Char('程序工时', readonly=True)
|
||||||
|
planned_start_date = fields.Date('计划开工日期')
|
||||||
|
actual_start_date = fields.Datetime('实际开工日期', compute='_compute_actual_start_date', store=True)
|
||||||
|
actual_end_date = fields.Datetime('实际完工日期', compute='_compute_actual_end_date', store=True)
|
||||||
|
print_count = fields.Char('打印次数', default='T0C0', readonly=True)
|
||||||
|
sequence = fields.Integer('序号')
|
||||||
|
|
||||||
|
hide_action_open_mrp_production = fields.Boolean(
|
||||||
|
string='显示待工艺确认按钮',
|
||||||
|
compute='_compute_hid_button',
|
||||||
|
default=False
|
||||||
|
)
|
||||||
|
|
||||||
|
hide_action_purchase_orders = fields.Boolean(
|
||||||
|
string='显示采购按钮',
|
||||||
|
compute='_compute_hide_action_purchase_orders',
|
||||||
|
default=False
|
||||||
|
)
|
||||||
|
|
||||||
|
hide_action_stock_picking = fields.Boolean(
|
||||||
|
string='显示调拨单按钮',
|
||||||
|
compute='_compute_hide_action_stock_picking',
|
||||||
|
default=False
|
||||||
|
)
|
||||||
|
|
||||||
|
hide_action_outsourcing_stock_picking = fields.Boolean(
|
||||||
|
string='委外显示调拨单按钮',
|
||||||
|
compute='_compute_hide_action_stock_picking',
|
||||||
|
default=False
|
||||||
|
)
|
||||||
|
|
||||||
|
hide_action_view_programming = fields.Boolean(
|
||||||
|
string='显示编程单按钮',
|
||||||
|
compute='_compute_hid_button',
|
||||||
|
default=False
|
||||||
|
)
|
||||||
|
|
||||||
|
outsourcing_purchase_request = fields.Char('委外采购申请单')
|
||||||
|
|
||||||
|
@api.depends('sale_order_id.state', 'sale_order_id.mrp_production_ids.schedule_state', 'sale_order_id.order_line',
|
||||||
|
'sale_order_id.mrp_production_ids.state')
|
||||||
|
def _compute_status(self):
|
||||||
|
for record in self:
|
||||||
|
if record.sale_order_id:
|
||||||
|
sale_order_state = record.sale_order_id.state
|
||||||
|
if sale_order_state in ('draft', 'sent', 'supply method'):
|
||||||
|
record.status = '20' # 待确认
|
||||||
|
if record.supply_method in ('purchase', 'outsourcing') and sale_order_state in (
|
||||||
|
'sale', 'processing', 'physical_distribution', 'delivered',
|
||||||
|
'done') and sale_order_state != 'cancel':
|
||||||
|
record.status = '60' # 已下达
|
||||||
|
if record.supply_method in ('automation', 'manual'):
|
||||||
|
if sale_order_state in (
|
||||||
|
'sale', 'processing', 'physical_distribution', 'delivered',
|
||||||
|
'done') and sale_order_state != 'cancel':
|
||||||
|
record.status = '30' # 需求确认
|
||||||
|
# 检查所有制造订单的排程单状态,有一个为待排程状态,就为待下达生产
|
||||||
|
pending_productions = record.sale_order_id.mrp_production_ids.filtered(
|
||||||
|
lambda p: p.state == 'confirmed' and p.product_id.id == record.product_id.id
|
||||||
|
)
|
||||||
|
if pending_productions:
|
||||||
|
record.status = '50' # 待下达生产
|
||||||
|
# 检查所有制造订单的排程单状态
|
||||||
|
if record.sale_order_id.mrp_production_ids:
|
||||||
|
product_productions = record.sale_order_id.mrp_production_ids.filtered(
|
||||||
|
lambda p: p.product_id.id == record.product_id.id
|
||||||
|
)
|
||||||
|
if product_productions and all(order.schedule_state != '未排' for order in product_productions):
|
||||||
|
record.status = '60' # 已下达
|
||||||
|
if sale_order_state == 'cancel' or not record.sale_order_line_id:
|
||||||
|
record.status = '100' # 取消
|
||||||
|
|
||||||
|
@api.depends('sale_order_line_id.product_id.name')
|
||||||
|
def _compute_sale_order_line_number(self):
|
||||||
|
for line in self:
|
||||||
|
if line.product_id:
|
||||||
|
line.sale_order_line_number = line.sale_order_line_id.product_id.name[-1]
|
||||||
|
else:
|
||||||
|
line.sale_order_line_number = None
|
||||||
|
|
||||||
|
@api.depends('product_id.part_number', 'product_id.model_name')
|
||||||
|
def _compute_part_number(self):
|
||||||
|
for line in self:
|
||||||
|
if line.product_id:
|
||||||
|
if line.product_id.part_number:
|
||||||
|
line.part_number = line.product_id.part_number
|
||||||
|
else:
|
||||||
|
if line.product_id.model_name:
|
||||||
|
line.part_number = line.product_id.model_name.rsplit('.', 1)[0]
|
||||||
|
else:
|
||||||
|
line.part_number = None
|
||||||
|
|
||||||
|
@api.depends('product_id.length', 'product_id.width', 'product_id.height')
|
||||||
|
def _compute_model_long(self):
|
||||||
|
for line in self:
|
||||||
|
if line.product_id:
|
||||||
|
line.model_long = f"{line.product_id.length}*{line.product_id.width}*{line.product_id.height}"
|
||||||
|
else:
|
||||||
|
line.model_long = None
|
||||||
|
|
||||||
|
@api.depends('product_id.model_long', 'product_id.model_width', 'product_id.model_height', 'product_id.blank_type')
|
||||||
|
def _compute_embryo_long(self):
|
||||||
|
for line in self:
|
||||||
|
if line.product_id:
|
||||||
|
if line.product_id.blank_type == '圆料':
|
||||||
|
line.embryo_long = f"Ø{round(line.product_id.model_width, 3)}*{round(line.product_id.model_long, 3)}"
|
||||||
|
else:
|
||||||
|
line.embryo_long = f"{round(line.product_id.model_long, 3)}*{round(line.product_id.model_width, 3)}*{round(line.product_id.model_height, 3)}"
|
||||||
|
else:
|
||||||
|
line.embryo_long = None
|
||||||
|
|
||||||
|
@api.depends('product_id.materials_id')
|
||||||
|
def _compute_materials_id(self):
|
||||||
|
for line in self:
|
||||||
|
if line.product_id:
|
||||||
|
line.materials_id = f"{line.product_id.materials_id.name}/{line.product_id.materials_type_id.name}"
|
||||||
|
else:
|
||||||
|
line.materials_id = None
|
||||||
|
|
||||||
|
@api.depends('product_id.model_process_parameters_ids')
|
||||||
|
def _compute_model_process_parameters_ids(self):
|
||||||
|
for line in self:
|
||||||
|
if line.product_id and line.product_id.model_process_parameters_ids:
|
||||||
|
line.model_process_parameters_ids = [(6, 0, line.product_id.model_process_parameters_ids.ids)]
|
||||||
|
else:
|
||||||
|
line.model_process_parameters_ids = [(5, 0, 0)]
|
||||||
|
|
||||||
|
def _compute_inventory_quantity_auto_apply(self):
|
||||||
|
location_id = self.env['stock.location'].search([('name', '=', '成品存货区')], limit=1).id
|
||||||
|
product_ids = self.mapped('product_id').ids
|
||||||
|
if product_ids:
|
||||||
|
quant_data = self.env['stock.quant'].read_group(
|
||||||
|
domain=[
|
||||||
|
('product_id', 'in', product_ids),
|
||||||
|
('location_id', '=', location_id)
|
||||||
|
],
|
||||||
|
fields=['product_id', 'inventory_quantity_auto_apply'],
|
||||||
|
groupby=['product_id']
|
||||||
|
)
|
||||||
|
quantity_map = {item['product_id'][0]: item['inventory_quantity_auto_apply'] for item in quant_data}
|
||||||
|
else:
|
||||||
|
quantity_map = {}
|
||||||
|
for line in self:
|
||||||
|
if line.product_id:
|
||||||
|
line.inventory_quantity_auto_apply = quantity_map.get(line.product_id.id, 0.0)
|
||||||
|
else:
|
||||||
|
line.inventory_quantity_auto_apply = 0.0
|
||||||
|
|
||||||
|
@api.depends('sale_order_id.mrp_production_ids.workorder_ids.date_start')
|
||||||
|
def _compute_actual_start_date(self):
|
||||||
|
for record in self:
|
||||||
|
if record.sale_order_id and record.sale_order_id.mrp_production_ids:
|
||||||
|
manufacturing_orders = record.sale_order_id.mrp_production_ids.filtered(
|
||||||
|
lambda mo: mo.product_id == record.product_id)
|
||||||
|
if manufacturing_orders:
|
||||||
|
start_dates = [
|
||||||
|
workorder.date_start for mo in manufacturing_orders
|
||||||
|
for workorder in mo.workorder_ids if workorder.date_start
|
||||||
|
]
|
||||||
|
record.actual_start_date = min(start_dates) if start_dates else None
|
||||||
|
else:
|
||||||
|
record.actual_start_date = None
|
||||||
|
else:
|
||||||
|
record.actual_start_date = None
|
||||||
|
|
||||||
|
@api.depends('sale_order_id.mrp_production_ids.workorder_ids.state',
|
||||||
|
'sale_order_id.mrp_production_ids.workorder_ids.date_finished')
|
||||||
|
def _compute_actual_end_date(self):
|
||||||
|
for record in self:
|
||||||
|
if record.sale_order_id and record.sale_order_id.mrp_production_ids:
|
||||||
|
manufacturing_orders = record.sale_order_id.mrp_production_ids.filtered(
|
||||||
|
lambda mo: mo.product_id == record.product_id)
|
||||||
|
finished_orders = manufacturing_orders.filtered(lambda mo: mo.state == 'done')
|
||||||
|
sum_product_qty = sum(finished_orders.mapped('product_qty'))
|
||||||
|
if finished_orders and float_compare(sum_product_qty, record.product_uom_qty,
|
||||||
|
precision_rounding=record.product_id.uom_id.rounding) >= 0:
|
||||||
|
end_dates = [
|
||||||
|
workorder.date_finished for mo in finished_orders
|
||||||
|
for workorder in mo.workorder_ids if workorder.date_finished
|
||||||
|
]
|
||||||
|
record.actual_end_date = max(end_dates) if end_dates else None
|
||||||
|
else:
|
||||||
|
record.actual_end_date = None
|
||||||
|
else:
|
||||||
|
record.actual_end_date = None
|
||||||
|
|
||||||
|
@api.depends('sale_order_id.mrp_production_ids.move_raw_ids.reserved_availability')
|
||||||
|
def _compute_material_check(self):
|
||||||
|
for record in self:
|
||||||
|
if record.sale_order_id and record.sale_order_id.mrp_production_ids:
|
||||||
|
manufacturing_orders = record.sale_order_id.mrp_production_ids.filtered(
|
||||||
|
lambda mo: mo.product_id == record.product_id)
|
||||||
|
|
||||||
|
if manufacturing_orders and manufacturing_orders.move_raw_ids:
|
||||||
|
# 获取完成的制造订单
|
||||||
|
done_manufacturing = manufacturing_orders.filtered(lambda mo: mo.state == 'done')
|
||||||
|
product_qty = sum(done_manufacturing.mapped('product_qty'))
|
||||||
|
# 需求数量-完成数量
|
||||||
|
product_uom_qty = record.product_uom_qty - product_qty
|
||||||
|
total_reserved_availability = sum(manufacturing_orders.mapped('move_raw_ids.reserved_availability'))
|
||||||
|
if float_compare(total_reserved_availability, product_uom_qty,
|
||||||
|
precision_rounding=record.product_id.uom_id.rounding) >= 0:
|
||||||
|
record.material_check = '1' # 已齐套
|
||||||
|
else:
|
||||||
|
record.material_check = '0' # 未齐套
|
||||||
|
else:
|
||||||
|
record.material_check = None
|
||||||
|
else:
|
||||||
|
record.material_check = None
|
||||||
|
|
||||||
|
@api.constrains('planned_start_date')
|
||||||
|
def _check_planned_start_date(self):
|
||||||
|
for record in self:
|
||||||
|
if record.planned_start_date and record.planned_start_date < fields.Date.today():
|
||||||
|
raise ValidationError("计划开工日期必须大于或等于今天。")
|
||||||
|
|
||||||
|
def release_production_order(self):
|
||||||
|
if not self.planned_start_date:
|
||||||
|
raise ValidationError("请先填写计划开工日期")
|
||||||
|
pro_plan_list = self.env['sf.production.plan'].search(
|
||||||
|
[('product_id', '=', self.product_id.id), ('state', '=', 'draft')])
|
||||||
|
sf_production_line = self.env['sf.production.line'].sudo().search(
|
||||||
|
[('name', '=', '1#CNC自动生产线')], limit=1)
|
||||||
|
if sf_production_line:
|
||||||
|
now = datetime.now()
|
||||||
|
time_part = (now + timedelta(hours=2)).time()
|
||||||
|
date_part = fields.Date.from_string(self.planned_start_date)
|
||||||
|
date_planned_start = datetime.combine(date_part, time_part)
|
||||||
|
pro_plan_list.production_line_id = sf_production_line.id
|
||||||
|
pro_plan_list.date_planned_start = date_planned_start
|
||||||
|
for pro_plan in pro_plan_list:
|
||||||
|
pro_plan.do_production_schedule()
|
||||||
|
|
||||||
|
def button_action_print(self):
|
||||||
|
return {
|
||||||
|
'res_model': 'sf.demand.plan.print.wizard',
|
||||||
|
'type': 'ir.actions.act_window',
|
||||||
|
'name': _("打印"),
|
||||||
|
'domain': [('demand_plan_id', 'in', self.ids)],
|
||||||
|
'views': [[self.env.ref('sf_demand_plan.action_plan_print_tree').id, 'list']],
|
||||||
|
'target': 'new',
|
||||||
|
}
|
||||||
|
|
||||||
|
@api.depends('sale_order_id.mrp_production_ids.state', 'sale_order_id.mrp_production_ids.programming_state')
|
||||||
|
def _compute_hid_button(self):
|
||||||
|
for record in self:
|
||||||
|
mrp_production_ids = record.sale_order_id.mrp_production_ids.filtered(
|
||||||
|
lambda p: p.state == 'technology_to_confirmed' and p.product_id.id == record.product_id.id
|
||||||
|
)
|
||||||
|
record.hide_action_open_mrp_production = bool(mrp_production_ids) and record.supply_method in (
|
||||||
|
'automation', 'manual')
|
||||||
|
programming_mrp_production_ids = record.sale_order_id.mrp_production_ids.filtered(
|
||||||
|
lambda p: p.programming_state == '编程中' and p.product_id.id == record.product_id.id
|
||||||
|
)
|
||||||
|
record.hide_action_view_programming = bool(programming_mrp_production_ids)
|
||||||
|
|
||||||
|
def _compute_hide_action_purchase_orders(self):
|
||||||
|
for record in self:
|
||||||
|
record.hide_action_purchase_orders = False
|
||||||
|
outsourcing_purchase_request = []
|
||||||
|
if record.supply_method in ('automation',
|
||||||
|
'manual') and record.material_check == '0' and not record.sale_order_line_id.is_incoming_material:
|
||||||
|
mrp_production = record.sale_order_id.mrp_production_ids.filtered(
|
||||||
|
lambda p: p.product_id.id == record.product_id.id
|
||||||
|
).sorted(key=lambda p: p.id)
|
||||||
|
if mrp_production:
|
||||||
|
raw_materials = mrp_production.mapped('move_raw_ids.product_id')
|
||||||
|
if raw_materials:
|
||||||
|
purchase_orders = self.env['purchase.order'].sudo().search([
|
||||||
|
('state', '=', 'purchase'),
|
||||||
|
('order_line.product_id', 'in', raw_materials.ids)
|
||||||
|
])
|
||||||
|
total_purchase_quantity = sum(
|
||||||
|
sum(
|
||||||
|
order.order_line.filtered(
|
||||||
|
lambda line: line.product_id in raw_materials
|
||||||
|
).mapped('product_qty')
|
||||||
|
)
|
||||||
|
for order in purchase_orders
|
||||||
|
)
|
||||||
|
if float_compare(total_purchase_quantity, record.product_uom_qty,
|
||||||
|
precision_rounding=record.product_id.uom_id.rounding) == -1:
|
||||||
|
pr_ids = self.env['purchase.request'].sudo().search(
|
||||||
|
[('line_ids.product_id', 'in', raw_materials.ids), ('state', '!=', 'done')])
|
||||||
|
outsourcing_purchase_request.extend(pr_ids.ids)
|
||||||
|
elif record.supply_method in ('purchase', 'outsourcing'):
|
||||||
|
purchase_orders = self.env['purchase.order'].sudo().search([
|
||||||
|
('state', 'in', ('purchase', 'done')),
|
||||||
|
('order_line.product_id', '=', record.product_id.id)
|
||||||
|
])
|
||||||
|
total_purchase_quantity = sum(
|
||||||
|
sum(
|
||||||
|
order.order_line.filtered(
|
||||||
|
lambda line: line.product_id in record.product_id
|
||||||
|
).mapped('product_qty')
|
||||||
|
)
|
||||||
|
for order in purchase_orders
|
||||||
|
)
|
||||||
|
|
||||||
|
if float_compare(total_purchase_quantity, record.product_uom_qty,
|
||||||
|
precision_rounding=record.product_id.uom_id.rounding) == -1:
|
||||||
|
pr_ids = self.env['purchase.request'].sudo().search(
|
||||||
|
[('origin', 'like', record.sale_order_id.name), ('state', '!=', 'done')])
|
||||||
|
outsourcing_purchase_request.extend(pr_ids.ids)
|
||||||
|
if record.supply_method == 'outsourcing' and not record.sale_order_line_id.is_incoming_material:
|
||||||
|
bom_line_ids = record.product_id.bom_ids.bom_line_ids
|
||||||
|
if bom_line_ids:
|
||||||
|
# BOM_数量
|
||||||
|
total_product_qty = sum(line.product_qty for line in bom_line_ids)
|
||||||
|
bom_product_ids = bom_line_ids.mapped('product_id')
|
||||||
|
product_purchase_orders = self.env['purchase.order'].sudo().search([
|
||||||
|
('state', 'in', ('purchase', 'done')),
|
||||||
|
('order_line.product_id', 'in', bom_product_ids.ids)
|
||||||
|
])
|
||||||
|
# 购订单_数量
|
||||||
|
total_outsourcing_purchase_quantity = sum(
|
||||||
|
sum(
|
||||||
|
order.order_line.filtered(
|
||||||
|
lambda line: line.product_id in bom_product_ids
|
||||||
|
).mapped('product_qty')
|
||||||
|
)
|
||||||
|
for order in product_purchase_orders
|
||||||
|
)
|
||||||
|
quantity = total_outsourcing_purchase_quantity / total_product_qty
|
||||||
|
if float_compare(quantity, record.product_uom_qty,
|
||||||
|
precision_rounding=record.product_id.uom_id.rounding) == -1:
|
||||||
|
purchase_request = self.env['purchase.request'].sudo().search(
|
||||||
|
[('line_ids.product_id', 'in', bom_product_ids.ids),
|
||||||
|
('line_ids.purchase_state', 'not in', ('purchase', 'done')), ('state', '!=', 'done')])
|
||||||
|
outsourcing_purchase_request.extend(purchase_request.ids)
|
||||||
|
record.outsourcing_purchase_request = json.dumps(outsourcing_purchase_request)
|
||||||
|
if outsourcing_purchase_request:
|
||||||
|
record.hide_action_purchase_orders = True
|
||||||
|
|
||||||
|
@api.depends('sale_order_id.mrp_production_ids.picking_ids.state', 'sale_order_id.picking_ids.state')
|
||||||
|
def _compute_hide_action_stock_picking(self):
|
||||||
|
for record in self:
|
||||||
|
record.hide_action_stock_picking = False
|
||||||
|
record.hide_action_outsourcing_stock_picking = False
|
||||||
|
if record.supply_method in ('automation', 'manual'):
|
||||||
|
manufacturing_orders = record.sale_order_id.mrp_production_ids.filtered(
|
||||||
|
lambda p: p.product_id.id == record.product_id.id
|
||||||
|
)
|
||||||
|
record.hide_action_stock_picking = bool(manufacturing_orders.mapped('picking_ids').filtered(
|
||||||
|
lambda p: p.state == 'assigned'))
|
||||||
|
elif record.supply_method in ('purchase', 'outsourcing'):
|
||||||
|
assigned_picking_ids = record.sale_order_id.picking_ids.filtered(
|
||||||
|
lambda
|
||||||
|
p: p.state == 'assigned' and p.picking_type_id.name != '发料出库' and p.move_line_ids.product_id in record.product_id)
|
||||||
|
if record.supply_method == 'outsourcing':
|
||||||
|
outsourcing_assigned_picking_ids = record.get_outsourcing_picking_ids()
|
||||||
|
record.hide_action_outsourcing_stock_picking = outsourcing_assigned_picking_ids
|
||||||
|
record.hide_action_stock_picking = assigned_picking_ids or outsourcing_assigned_picking_ids
|
||||||
|
else:
|
||||||
|
record.hide_action_stock_picking = assigned_picking_ids
|
||||||
|
|
||||||
|
def get_outsourcing_picking_ids(self):
|
||||||
|
order_ids = self.env['purchase.order'].sudo().search(
|
||||||
|
[('order_line.product_id', 'in', self.product_id.ids),
|
||||||
|
('purchase_type', '=', 'outsourcing')])
|
||||||
|
outsourcing_picking_ids = order_ids._get_subcontracting_resupplies()
|
||||||
|
outsourcing_assigned_picking_ids = outsourcing_picking_ids.filtered(lambda p: p.state == 'assigned')
|
||||||
|
return outsourcing_assigned_picking_ids
|
||||||
|
|
||||||
|
def action_open_sale_order(self):
|
||||||
|
self.ensure_one()
|
||||||
|
return {
|
||||||
|
'type': 'ir.actions.act_window',
|
||||||
|
'res_model': 'sale.order',
|
||||||
|
'res_id': self.sale_order_id.id,
|
||||||
|
'view_mode': 'form',
|
||||||
|
}
|
||||||
|
|
||||||
|
def action_open_mrp_production(self):
|
||||||
|
self.ensure_one()
|
||||||
|
mrp_production_ids = self.sale_order_id.mrp_production_ids.filtered(
|
||||||
|
lambda p: p.state == 'technology_to_confirmed' and p.product_id.id == self.product_id.id
|
||||||
|
)
|
||||||
|
action = {
|
||||||
|
'res_model': 'mrp.production',
|
||||||
|
'type': 'ir.actions.act_window',
|
||||||
|
}
|
||||||
|
if len(mrp_production_ids) == 1:
|
||||||
|
action.update({
|
||||||
|
'view_mode': 'form',
|
||||||
|
'res_id': mrp_production_ids.id,
|
||||||
|
})
|
||||||
|
else:
|
||||||
|
action.update({
|
||||||
|
'name': _("制造订单列表"),
|
||||||
|
'domain': [('id', 'in', mrp_production_ids.ids)],
|
||||||
|
'view_mode': 'tree,form',
|
||||||
|
})
|
||||||
|
return action
|
||||||
|
|
||||||
|
def action_view_purchase_request(self):
|
||||||
|
self.ensure_one()
|
||||||
|
pr_ids = self.env['purchase.request'].sudo().search(
|
||||||
|
[('id', 'in', ast.literal_eval(self.outsourcing_purchase_request))])
|
||||||
|
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': _("采购申请"),
|
||||||
|
'domain': [('id', 'in', pr_ids.ids)],
|
||||||
|
'view_mode': 'tree,form',
|
||||||
|
})
|
||||||
|
return action
|
||||||
|
|
||||||
|
def action_view_stock_picking(self):
|
||||||
|
self.ensure_one()
|
||||||
|
action = self.env["ir.actions.actions"]._for_xml_id("stock.action_picking_tree_all")
|
||||||
|
picking_ids = None
|
||||||
|
if self.supply_method in ('automation', 'manual'):
|
||||||
|
mrp_production_ids = self.sale_order_id.mrp_production_ids.filtered(
|
||||||
|
lambda p: p.product_id.id == self.product_id.id
|
||||||
|
)
|
||||||
|
picking_ids = mrp_production_ids.mapped('picking_ids').filtered(
|
||||||
|
lambda p: p.state == 'assigned')
|
||||||
|
elif self.supply_method in ('purchase', 'outsourcing'):
|
||||||
|
picking_ids = self.sale_order_id.picking_ids.filtered(
|
||||||
|
lambda
|
||||||
|
p: p.state == 'assigned' and p.picking_type_id.name != '发料出库' and p.move_line_ids.product_id in self.product_id)
|
||||||
|
if self.supply_method == 'outsourcing' and self.hide_action_outsourcing_stock_picking:
|
||||||
|
picking_ids = picking_ids.union(self.get_outsourcing_picking_ids())
|
||||||
|
if picking_ids:
|
||||||
|
if len(picking_ids) > 1:
|
||||||
|
action['domain'] = [('id', 'in', picking_ids.ids)]
|
||||||
|
elif picking_ids:
|
||||||
|
action['res_id'] = picking_ids.id
|
||||||
|
action['views'] = [(self.env.ref('stock.view_picking_form').id, 'form')]
|
||||||
|
if 'views' in action:
|
||||||
|
action['views'] += [(state, view) for state, view in action['views'] if view != 'form']
|
||||||
|
return action
|
||||||
|
|
||||||
|
def action_view_programming(self):
|
||||||
|
self.ensure_one()
|
||||||
|
programming_mrp_production_ids = self.sale_order_id.mrp_production_ids.filtered(
|
||||||
|
lambda p: p.programming_state == '编程中' and p.product_id.id == self.product_id.id
|
||||||
|
).mapped('programming_no')
|
||||||
|
if programming_mrp_production_ids:
|
||||||
|
programming_no = list(set(programming_mrp_production_ids))
|
||||||
|
numbers_str = "、".join(programming_no)
|
||||||
|
raise ValidationError(f"编程单号:{numbers_str},请去云平台处理")
|
||||||
6
sf_demand_plan/security/ir.model.access.csv
Normal file
6
sf_demand_plan/security/ir.model.access.csv
Normal file
@@ -0,0 +1,6 @@
|
|||||||
|
id,name,model_id:id,group_id:id,perm_read,perm_write,perm_create,perm_unlink
|
||||||
|
access_sf_production_demand_plan,sf.production.demand.plan,model_sf_production_demand_plan,base.group_user,1,0,0,0
|
||||||
|
access_sf_production_demand_plan_for_dispatch,sf.production.demand.plan for dispatch,model_sf_production_demand_plan,sf_base.group_plan_dispatch,1,1,0,0
|
||||||
|
|
||||||
|
access_sf_demand_plan_print_wizard,sf.demand.plan.print.wizard,model_sf_demand_plan_print_wizard,base.group_user,1,0,0,0
|
||||||
|
access_sf_demand_plan_print_wizard_for_dispatch,sf.demand.plan.print.wizard for dispatch,model_sf_demand_plan_print_wizard,sf_base.group_plan_dispatch,1,1,0,0
|
||||||
|
256
sf_demand_plan/static/src/js/print_demand.js
Normal file
256
sf_demand_plan/static/src/js/print_demand.js
Normal file
@@ -0,0 +1,256 @@
|
|||||||
|
odoo.define('sf_demand.print_demand', function (require) {
|
||||||
|
"use strict";
|
||||||
|
|
||||||
|
var ListController = require('web.ListController');
|
||||||
|
var ListRenderer = require('web.ListRenderer');
|
||||||
|
var ListView = require('web.ListView');
|
||||||
|
var viewRegistry = require('web.view_registry');
|
||||||
|
var { url } = require("@web/core/utils/urls")
|
||||||
|
|
||||||
|
var CustomListRenderer = ListRenderer.extend({
|
||||||
|
_render: function () {
|
||||||
|
var self = this;
|
||||||
|
this.getParent()?.$buttons.hide();
|
||||||
|
|
||||||
|
return this._super.apply(this, arguments).then(function () {
|
||||||
|
if(!self.state.data || !self.state.data.length) return
|
||||||
|
// 添加图片预览容器到页面左侧
|
||||||
|
if (!$('.table-image-preview-container').length) {
|
||||||
|
self.$el.parent().addClass('custom-table-image-container')
|
||||||
|
self.$el.before(
|
||||||
|
`<div class="custom-preview-container">
|
||||||
|
<img class="table-image-preview-container" src="" />
|
||||||
|
<iframe class="table-image-preview-container" src=""/>
|
||||||
|
</div>`
|
||||||
|
);
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
if(!$('.denmand_set').length) {
|
||||||
|
|
||||||
|
const checked = self.getParent().radioCheck || 'all'
|
||||||
|
|
||||||
|
self.$el.prepend(`
|
||||||
|
<form class="denmand_set">
|
||||||
|
<span>更多设置:</span>
|
||||||
|
<input type="radio" id="male" name="set" value="图纸">
|
||||||
|
<label for="male">图纸</label>
|
||||||
|
<input type="radio" id="female" name="set" value="程序单">
|
||||||
|
<label for="female">程序单</label>
|
||||||
|
<input type="radio" id="other" name="set" value="all" >
|
||||||
|
<label for="other">图纸/程序单</label>
|
||||||
|
</form>
|
||||||
|
`)
|
||||||
|
setTimeout(() => {
|
||||||
|
$(`input[name=set][value=${checked}]`).prop('checked', true)
|
||||||
|
$('.denmand_set').trigger('click')
|
||||||
|
}, 100);
|
||||||
|
self.$el.prepend(`
|
||||||
|
<div class="print-button-container" style="margin-bottom:10px;">
|
||||||
|
<button class="btn btn-primary o_print_custom">
|
||||||
|
<i class="fa fa-print"></i> 打印
|
||||||
|
</button>
|
||||||
|
<button class="btn btn-secondary o_cancel_custom">
|
||||||
|
取消
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
`);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
},
|
||||||
|
start: function() {
|
||||||
|
setTimeout(() => {
|
||||||
|
this.$el.find('.o_data_row').eq(0).trigger('click')
|
||||||
|
this.getParent().$el?.find('.o_cp_top_right,.o_cp_bottom').hide()
|
||||||
|
}, 500);
|
||||||
|
return this._super();
|
||||||
|
},
|
||||||
|
events: _.extend({}, ListRenderer.prototype.events, {
|
||||||
|
'click .o_data_row': '_onCustomRowClick',
|
||||||
|
'click .o_print_custom': '_onPrintClick',
|
||||||
|
'click .o_cancel_custom': '_onCancelClick',
|
||||||
|
'click .denmand_set': '_onDenmandChange',
|
||||||
|
}),
|
||||||
|
_onCancelClick() {
|
||||||
|
this.getParent()?.getParent()?.dialogs.closeAll()
|
||||||
|
},
|
||||||
|
_onCustomRowClick: async function (ev) {
|
||||||
|
var self = this;
|
||||||
|
var $row = $(ev.currentTarget);
|
||||||
|
var index = $row.index();
|
||||||
|
var data = this.state.data[index];
|
||||||
|
if(data.fileData?.fileUrl) {
|
||||||
|
|
||||||
|
} else {
|
||||||
|
data.fileData = { }
|
||||||
|
if(data.res_id) {
|
||||||
|
// 正确获取 ORM 服务的方式
|
||||||
|
// var orm = this.getParent().getParent().env.services
|
||||||
|
const key = data.data.type == 1 ? 'machining_drawings' : 'cnc_worksheet'
|
||||||
|
const attachment = await this._rpc({
|
||||||
|
model: 'ir.binary',
|
||||||
|
method: 'attachment_info',
|
||||||
|
args: [
|
||||||
|
data.model,
|
||||||
|
data.res_id,
|
||||||
|
key
|
||||||
|
],
|
||||||
|
})
|
||||||
|
|
||||||
|
if (attachment) {
|
||||||
|
Object.assign(data.fileData, attachment)
|
||||||
|
this._getTypeInfo(attachment.mimetype, data.fileData)
|
||||||
|
}
|
||||||
|
const fileUrl = this.getFileUrl(data.fileData.attachment_type, data, key)
|
||||||
|
data.fileData.fileUrl = fileUrl
|
||||||
|
}
|
||||||
|
}
|
||||||
|
$('.table-image-preview-container').hide()
|
||||||
|
if(data.fileData.attachment_type == 'iframe') {
|
||||||
|
|
||||||
|
$('iframe.table-image-preview-container').attr('src', decodeURIComponent(data.fileData.fileUrl) ).show()
|
||||||
|
} else {
|
||||||
|
$('img.table-image-preview-container').attr('src', data.fileData.fileUrl).show()
|
||||||
|
}
|
||||||
|
},
|
||||||
|
getSelectedIds: function() {
|
||||||
|
return this.state.data.filter(_ => !_.hide).map(_ => {
|
||||||
|
return _.data.id
|
||||||
|
})
|
||||||
|
},
|
||||||
|
_onPrintClick(e) {
|
||||||
|
var print_ids = this.getSelectedIds();
|
||||||
|
this._rpc({
|
||||||
|
model: 'sf.demand.plan.print.wizard',
|
||||||
|
method: 'demand_plan_print',
|
||||||
|
args: [ print_ids ] ,
|
||||||
|
// context: this.state.getContext()
|
||||||
|
}).then((e) => {
|
||||||
|
|
||||||
|
this.getParent()?.getParent()?.env.services?.notification.notify( {
|
||||||
|
type: 'info',
|
||||||
|
message: e.message,
|
||||||
|
})
|
||||||
|
// self.do_notify("成功", "打印任务已发送到打印机");
|
||||||
|
}).catch(function(error) {
|
||||||
|
console.error("打印错误:", error);
|
||||||
|
// self.do_warn("打印失败", error.data.message || "未知错误");
|
||||||
|
}).finally(e => {
|
||||||
|
this.getParent().reload()
|
||||||
|
})
|
||||||
|
|
||||||
|
},
|
||||||
|
_onDenmandChange(e) {
|
||||||
|
const isChecked = $(e.currentTarget).find('input:checked').val()
|
||||||
|
this.getParent().radioCheck = isChecked
|
||||||
|
this.$el.find('tbody').find('.o_data_row').show()
|
||||||
|
|
||||||
|
this.state.data.forEach(_ => {
|
||||||
|
_.hide = false
|
||||||
|
})
|
||||||
|
const self = this
|
||||||
|
if(!isChecked || isChecked == 'all') return
|
||||||
|
this.$el.find('tbody').children('.o_data_row').each(function() {
|
||||||
|
if($(this).find('td[name=type]').text() != isChecked) {
|
||||||
|
const i = $(this).index()
|
||||||
|
|
||||||
|
self.state.data[i].hide = true
|
||||||
|
$(this).hide()
|
||||||
|
}
|
||||||
|
})
|
||||||
|
},
|
||||||
|
getFileUrl(attachment_type, data, key) {
|
||||||
|
let fileUrl
|
||||||
|
switch (attachment_type) {
|
||||||
|
case 'image':
|
||||||
|
const timer = +new Date()
|
||||||
|
fileUrl = url("/web/image", {
|
||||||
|
model: data.model,
|
||||||
|
id: data.res_id,
|
||||||
|
field: key,
|
||||||
|
unique: '',
|
||||||
|
timer
|
||||||
|
});
|
||||||
|
|
||||||
|
break;
|
||||||
|
case 'iframe':
|
||||||
|
|
||||||
|
const iframe_file_url = encodeURIComponent(
|
||||||
|
url("/web/content", {
|
||||||
|
model: data.model,
|
||||||
|
id: data.res_id,
|
||||||
|
field: key,
|
||||||
|
})
|
||||||
|
);
|
||||||
|
fileUrl = `${this.state.attachment_base}?file=${iframe_file_url}`;
|
||||||
|
case 'unknown':
|
||||||
|
fileUrl = encodeURIComponent(
|
||||||
|
url("/web/content", {
|
||||||
|
model: data.model,
|
||||||
|
id: data.res_id,
|
||||||
|
field: key,
|
||||||
|
})
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
return fileUrl
|
||||||
|
},
|
||||||
|
_getTypeInfo(type, data) {
|
||||||
|
switch (type) {
|
||||||
|
case 'application/pdf':
|
||||||
|
data.attachment_base = `/web/static/lib/pdfjs/web/viewer.html`;
|
||||||
|
data.attachment_type = 'iframe'
|
||||||
|
break;
|
||||||
|
case 'application/vnd.openxmlformats-officedocument.wordprocessingml.document':
|
||||||
|
data.attachment_base = `/jikimo_attachment_viewer/static/lib/docxjs/viewer.html`;
|
||||||
|
data.attachment_type = 'iframe'
|
||||||
|
break;
|
||||||
|
case 'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet':
|
||||||
|
data.attachment_base = `/jikimo_attachment_viewer/static/lib/exceljs/viewer.html`;
|
||||||
|
data.attachment_type = 'iframe'
|
||||||
|
break;
|
||||||
|
case 'application/vnd.openxmlformats-officedocument.presentationml.presentation':
|
||||||
|
data.attachment_base = '';
|
||||||
|
data.attachment_type = 'unknown'
|
||||||
|
break;
|
||||||
|
case 'image/png':
|
||||||
|
case 'image/jpeg':
|
||||||
|
case 'image/jpg':
|
||||||
|
data.attachment_base = ''
|
||||||
|
data.attachment_type = 'image'
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
data.attachment_base = ''
|
||||||
|
data.attachment_type = 'unknown'
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
var CustomListController = ListController.extend({
|
||||||
|
// 可以保留或移除,根据是否需要处理自定义事件
|
||||||
|
// 处理打印操作
|
||||||
|
// rpc.query({
|
||||||
|
// model: 'sf.demand.plan.print.wizard',
|
||||||
|
// method: 'demand_plan_print',
|
||||||
|
// args: [recordIds]
|
||||||
|
// }).then(function(result) {
|
||||||
|
// self.do_notify("成功", "打印任务已发送到打印机");
|
||||||
|
// // 刷新视图显示最新状态
|
||||||
|
// self.reload();
|
||||||
|
// }).catch(function(error) {
|
||||||
|
// self.do_warn("打印失败", error.data.message || "发生未知错误");
|
||||||
|
// });
|
||||||
|
});
|
||||||
|
|
||||||
|
var PrintDemand = ListView.extend({
|
||||||
|
config: _.extend({}, ListView.prototype.config, {
|
||||||
|
Renderer: CustomListRenderer,
|
||||||
|
Controller: CustomListController,
|
||||||
|
}),
|
||||||
|
});
|
||||||
|
|
||||||
|
viewRegistry.add('print_demand', PrintDemand);
|
||||||
|
|
||||||
|
return PrintDemand;
|
||||||
|
});
|
||||||
89
sf_demand_plan/static/src/scss/style.css
Normal file
89
sf_demand_plan/static/src/scss/style.css
Normal file
@@ -0,0 +1,89 @@
|
|||||||
|
.demand_plan_tree .o_list_table_ungrouped th:not(.o_list_record_selector,.row_no,[data-name=sequence]) {
|
||||||
|
min-width: 98px !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
.demand_plan_tree .o_list_table_grouped th:not(.o_list_record_selector,.row_no,[data-name=sequence]) {
|
||||||
|
width: 98px !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
.demand_plan_tree .o_list_table_ungrouped {
|
||||||
|
min-width: 1900px;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
.o_selected_row {
|
||||||
|
background-color: #e6f7ff !important;
|
||||||
|
font-weight: bold;
|
||||||
|
}
|
||||||
|
|
||||||
|
.custom-table-image-container {
|
||||||
|
display: flex;
|
||||||
|
height: calc(95vh - 254px);
|
||||||
|
gap: 20px;
|
||||||
|
position: relative;
|
||||||
|
th.o_list_record_selector, td.o_list_record_selector{
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
.custom-preview-container, .print_demand {
|
||||||
|
flex: 1;
|
||||||
|
|
||||||
|
max-width: 49%;
|
||||||
|
tbody {
|
||||||
|
tr:not(.o_data_row) {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
tfoot {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.print_demand {
|
||||||
|
.table-responsive {
|
||||||
|
width: 100%;
|
||||||
|
overflow-x: auto!important;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.custom-preview-container {
|
||||||
|
background-color: #dadce0;
|
||||||
|
padding: 20px;
|
||||||
|
img {
|
||||||
|
max-width: 100%;
|
||||||
|
max-height: 100%;
|
||||||
|
}
|
||||||
|
iframe {
|
||||||
|
width: 100%;
|
||||||
|
height: 100%;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.o_print_custom, .o_cancel_custom {
|
||||||
|
position: absolute;
|
||||||
|
bottom: 20px;
|
||||||
|
right: 20px;
|
||||||
|
}
|
||||||
|
.o_print_custom {
|
||||||
|
right: 66px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.denmand_set {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
height: 50px;
|
||||||
|
> span {
|
||||||
|
font-weight: bold;
|
||||||
|
}
|
||||||
|
input {
|
||||||
|
margin-left: 30px;
|
||||||
|
}
|
||||||
|
label {
|
||||||
|
margin-left: 5px;
|
||||||
|
}
|
||||||
|
input,label {
|
||||||
|
cursor: pointer;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
th[data-name=processing_time] + th::before{
|
||||||
|
content: '待执行单据';
|
||||||
|
line-height: 38px;
|
||||||
|
}
|
||||||
135
sf_demand_plan/views/demand_plan.xml
Normal file
135
sf_demand_plan/views/demand_plan.xml
Normal file
@@ -0,0 +1,135 @@
|
|||||||
|
<odoo>
|
||||||
|
<record id="view_sf_production_demand_plan_tree" model="ir.ui.view">
|
||||||
|
<field name="name">sf.production.demand.plan.tree</field>
|
||||||
|
<field name="model">sf.production.demand.plan</field>
|
||||||
|
<field name="arch" type="xml">
|
||||||
|
<tree string="需求计划" default_order="sequence desc,id desc" editable="bottom"
|
||||||
|
class="demand_plan_tree freeze-columns-before-part_number">
|
||||||
|
<header>
|
||||||
|
<button string="打印" name="button_action_print" type="object"
|
||||||
|
class="btn-primary"/>
|
||||||
|
</header>
|
||||||
|
<field name="sequence" widget="handle"/>
|
||||||
|
<field name="id" optional="hide"/>
|
||||||
|
<field name="status"/>
|
||||||
|
<field name="customer_name"/>
|
||||||
|
<field name="order_remark"/>
|
||||||
|
<field name="glb_url" optional="hide"/>
|
||||||
|
<field name="product_id"/>
|
||||||
|
<field name="model_id" optional="hide"/>
|
||||||
|
<field name="part_name"/>
|
||||||
|
<field name="part_number"/>
|
||||||
|
<field name="is_incoming_material"/>
|
||||||
|
<field name="supply_method"/>
|
||||||
|
<field name="product_uom_qty"/>
|
||||||
|
<field name="deadline_of_delivery"/>
|
||||||
|
<field name="inventory_quantity_auto_apply"/>
|
||||||
|
<field name="qty_delivered"/>
|
||||||
|
<field name="qty_to_deliver"/>
|
||||||
|
<field name="model_long"/>
|
||||||
|
<field name="blank_type" optional="hide"/>
|
||||||
|
<field name="blank_precision"/>
|
||||||
|
<field name="embryo_long"/>
|
||||||
|
<field name="materials_id"/>
|
||||||
|
<field name="model_machining_precision"/>
|
||||||
|
<field name="model_process_parameters_ids" widget="many2many_tags"/>
|
||||||
|
<field name="product_remark" optional="hide"/>
|
||||||
|
<field name="order_code" optional="hide"/>
|
||||||
|
<field name="sale_order_id" optional="hide"/>
|
||||||
|
<field name="sale_order_line_number" optional="hide"/>
|
||||||
|
<field name="order_state"/>
|
||||||
|
<field name="route_id" optional="hide"/>
|
||||||
|
<field name="contract_date"/>
|
||||||
|
<field name="date_order"/>
|
||||||
|
<field name="contract_code"/>
|
||||||
|
<field name="plan_remark"/>
|
||||||
|
<field name="priority" decoration-danger="priority == '1'"
|
||||||
|
decoration-warning="priority == '2'"
|
||||||
|
decoration-info="priority == '3'"
|
||||||
|
decoration-success="priority == '4'"/>
|
||||||
|
<field name="material_check" optional="hide"/>
|
||||||
|
<field name="hide_action_open_mrp_production" invisible="1"/>
|
||||||
|
<field name="hide_action_purchase_orders" invisible="1"/>
|
||||||
|
<field name="hide_action_stock_picking" invisible="1"/>
|
||||||
|
<field name="hide_action_view_programming" invisible="1"/>
|
||||||
|
<button name="action_open_sale_order" type="object" string="供货方式待确认" class="btn-secondary"
|
||||||
|
attrs="{'invisible': [('supply_method', '!=', False)]}"/>
|
||||||
|
<button name="action_open_mrp_production" type="object" string="待工艺确认" class="btn-secondary"
|
||||||
|
attrs="{'invisible': [('hide_action_open_mrp_production', '=', False)]}"/>
|
||||||
|
<button name="action_view_purchase_request" type="object" string="采购申请" class="btn-secondary"
|
||||||
|
attrs="{'invisible': [('hide_action_purchase_orders', '=', False)]}"/>
|
||||||
|
<button name="action_view_stock_picking" type="object" string="调拨单" class="btn-secondary"
|
||||||
|
attrs="{'invisible': [('hide_action_stock_picking', '=', False)]}"/>
|
||||||
|
<button name="action_view_programming" type="object" string="编程单" class="btn-secondary"
|
||||||
|
attrs="{'invisible': [('hide_action_view_programming', '=', False)]}"/>
|
||||||
|
<field name="planned_start_date"/>
|
||||||
|
<field name="actual_start_date"/>
|
||||||
|
<field name="actual_end_date"/>
|
||||||
|
<field name="processing_time"/>
|
||||||
|
<field name="create_date" optional="hide" string="创建时间"/>
|
||||||
|
<field name="create_uid" optional="hide" string="创建人"/>
|
||||||
|
<field name="write_date" string="更新时间"/>
|
||||||
|
<field name="write_uid" optional="hide" string="更新人"/>
|
||||||
|
<field name="print_count"/>
|
||||||
|
<button name="release_production_order" type="object" string="下达生产" class="btn-primary"
|
||||||
|
attrs="{'invisible': ['|',('status', '!=', '50'), ('supply_method', 'not in', ['automation', 'manual'])]}"/>
|
||||||
|
</tree>
|
||||||
|
</field>
|
||||||
|
</record>
|
||||||
|
|
||||||
|
<record id="view_sf_production_demand_plan_search" model="ir.ui.view">
|
||||||
|
<field name="name">sf.production.demand.plan.search</field>
|
||||||
|
<field name="model">sf.production.demand.plan</field>
|
||||||
|
<field name="arch" type="xml">
|
||||||
|
<search>
|
||||||
|
<field name="order_remark"/>
|
||||||
|
<field name="product_id"/>
|
||||||
|
<field name="part_name"/>
|
||||||
|
<field name="part_number"/>
|
||||||
|
<field name="customer_name"/>
|
||||||
|
<field name="supply_method"/>
|
||||||
|
<field name="materials_id"/>
|
||||||
|
<field name="model_process_parameters_ids"/>
|
||||||
|
<field name="plan_remark"/>
|
||||||
|
<field name="contract_code"/>
|
||||||
|
<group expand="0" string="Group By">
|
||||||
|
<filter name="group_by_priority" string="优先级" domain="[]" context="{'group_by': 'priority'}"/>
|
||||||
|
<filter name="group_by_status" string="状态" domain="[]" context="{'group_by': 'status'}"/>
|
||||||
|
<filter name="group_by_customer_name" string="客户" domain="[]"
|
||||||
|
context="{'group_by': 'customer_name'}"/>
|
||||||
|
<filter name="group_by_is_incoming_material" string="客供料" domain="[]"
|
||||||
|
context="{'group_by': 'is_incoming_material'}"/>
|
||||||
|
<filter name="group_by_supply_method" string="供货方式" domain="[]"
|
||||||
|
context="{'group_by': 'supply_method'}"/>
|
||||||
|
<filter name="group_by_deadline_of_delivery" string="客户交期" domain="[]"
|
||||||
|
context="{'group_by': 'deadline_of_delivery'}"/>
|
||||||
|
<filter name="group_by_materials_id" string="材料" domain="[]"
|
||||||
|
context="{'group_by': 'materials_id'}"/>
|
||||||
|
<filter name="group_by_contract_code" string="合同号" domain="[]"
|
||||||
|
context="{'group_by': 'contract_code'}"/>
|
||||||
|
</group>
|
||||||
|
</search>
|
||||||
|
</field>
|
||||||
|
</record>
|
||||||
|
|
||||||
|
<record id="sf_production_demand_plan_action" model="ir.actions.act_window">
|
||||||
|
<field name="name">需求计划</field>
|
||||||
|
<field name="type">ir.actions.act_window</field>
|
||||||
|
<field name="res_model">sf.production.demand.plan</field>
|
||||||
|
<field name="view_mode">tree</field>
|
||||||
|
</record>
|
||||||
|
|
||||||
|
|
||||||
|
<menuitem
|
||||||
|
id="demand_plan_menu"
|
||||||
|
name="需求计划"
|
||||||
|
sequence="140"
|
||||||
|
action="sf_production_demand_plan_action"
|
||||||
|
parent="sf_plan.sf_production_plan_menu"
|
||||||
|
/>
|
||||||
|
|
||||||
|
<!-- 调拨动作中屏蔽验证-->
|
||||||
|
<record id="stock.action_validate_picking" model="ir.actions.server">
|
||||||
|
<field name="binding_model_id" eval="False"/>
|
||||||
|
</record>
|
||||||
|
</odoo>
|
||||||
1
sf_demand_plan/wizard/__init__.py
Normal file
1
sf_demand_plan/wizard/__init__.py
Normal file
@@ -0,0 +1 @@
|
|||||||
|
from . import sf_demand_plan_print_wizard
|
||||||
95
sf_demand_plan/wizard/sf_demand_plan_print_wizard.py
Normal file
95
sf_demand_plan/wizard/sf_demand_plan_print_wizard.py
Normal file
@@ -0,0 +1,95 @@
|
|||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
import logging
|
||||||
|
from odoo import models, fields, api, _
|
||||||
|
|
||||||
|
_logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
|
class SfDemandPlanPrintWizard(models.TransientModel):
|
||||||
|
_name = 'sf.demand.plan.print.wizard'
|
||||||
|
_description = u'打印向导'
|
||||||
|
|
||||||
|
demand_plan_id = fields.Many2one('sf.production.demand.plan', string='需求计划ID')
|
||||||
|
product_id = fields.Many2one(
|
||||||
|
comodel_name='product.product',
|
||||||
|
related='demand_plan_id.product_id',
|
||||||
|
string='产品', store=True, index=True)
|
||||||
|
model_id = fields.Char('模型ID')
|
||||||
|
filename_url = fields.Char('文件名/URL')
|
||||||
|
type = fields.Selection([
|
||||||
|
('1', '图纸'),
|
||||||
|
('2', '程序单'),
|
||||||
|
], string='类型')
|
||||||
|
status = fields.Selection([
|
||||||
|
('not_start', '未开始'),
|
||||||
|
('success', '成功'),
|
||||||
|
('fail', '失败'),
|
||||||
|
], string='状态', default='not_start')
|
||||||
|
machining_drawings = fields.Binary('2D加工图纸', related='product_id.machining_drawings', store=True)
|
||||||
|
|
||||||
|
cnc_worksheet = fields.Binary('程序单')
|
||||||
|
|
||||||
|
@api.model
|
||||||
|
def demand_plan_print(self, print_ids):
|
||||||
|
plan_print_ids = self.env['sf.demand.plan.print.wizard'].sudo().search(
|
||||||
|
[('id', 'in', print_ids)])
|
||||||
|
if not plan_print_ids:
|
||||||
|
return {'message': '记录不存在'}
|
||||||
|
success_records = []
|
||||||
|
failed_records = []
|
||||||
|
for record in plan_print_ids:
|
||||||
|
pdf_data = record.machining_drawings if record.type == '1' else record.cnc_worksheet
|
||||||
|
if pdf_data:
|
||||||
|
try:
|
||||||
|
# 执行打印
|
||||||
|
self.env['jikimo.printing'].sudo().print_pdf(pdf_data)
|
||||||
|
record.status = 'success'
|
||||||
|
t_part, c_part = record.demand_plan_id.print_count.split('C')
|
||||||
|
t_num = int(t_part[1:])
|
||||||
|
c_num = int(c_part)
|
||||||
|
if record.type == '1':
|
||||||
|
t_num += 1
|
||||||
|
elif record.type == '2':
|
||||||
|
c_num += 1
|
||||||
|
record.demand_plan_id.print_count = f"T{t_num}C{c_num}"
|
||||||
|
success_records.append({
|
||||||
|
'filename_url': record.filename_url,
|
||||||
|
})
|
||||||
|
except Exception as e:
|
||||||
|
record.status = 'fail'
|
||||||
|
_logger.error(f"文件{record.filename_url}打印失败: {str(e)}")
|
||||||
|
failed_records.append({
|
||||||
|
'filename_url': record.filename_url,
|
||||||
|
})
|
||||||
|
if failed_records:
|
||||||
|
message = f"成功打印 {len(success_records)} 个文件,失败 {len(failed_records)} 个"
|
||||||
|
else:
|
||||||
|
message = f"所有 {len(success_records)} 个文件打印成功"
|
||||||
|
return {'message': message}
|
||||||
|
|
||||||
|
|
||||||
|
class MrpWorkorder(models.Model):
|
||||||
|
_inherit = 'mrp.workorder'
|
||||||
|
|
||||||
|
def write(self, vals):
|
||||||
|
res = super(MrpWorkorder, self).write(vals)
|
||||||
|
for record in self:
|
||||||
|
if 'cnc_worksheet' in vals:
|
||||||
|
demand_plan_print = self.env['sf.demand.plan.print.wizard'].sudo().search(
|
||||||
|
[('model_id', '=', record.model_id), ('type', '=', '2')])
|
||||||
|
if demand_plan_print:
|
||||||
|
self.env['sf.demand.plan.print.wizard'].sudo().write(
|
||||||
|
{'cnc_worksheet': record.cnc_worksheet, 'filename_url': record.cnc_worksheet_name})
|
||||||
|
else:
|
||||||
|
demand_plan = self.env['sf.production.demand.plan'].sudo().search(
|
||||||
|
[('product_id', '=', record.product_id.id)])
|
||||||
|
if demand_plan:
|
||||||
|
wizard_vals = {
|
||||||
|
'demand_plan_id': demand_plan.id,
|
||||||
|
'model_id': demand_plan.model_id,
|
||||||
|
'type': '2',
|
||||||
|
'cnc_worksheet': record.cnc_worksheet,
|
||||||
|
'filename_url': record.cnc_worksheet_name
|
||||||
|
}
|
||||||
|
self.env['sf.demand.plan.print.wizard'].sudo().create(wizard_vals)
|
||||||
|
return res
|
||||||
17
sf_demand_plan/wizard/sf_demand_plan_print_wizard_view.xml
Normal file
17
sf_demand_plan/wizard/sf_demand_plan_print_wizard_view.xml
Normal file
@@ -0,0 +1,17 @@
|
|||||||
|
<?xml version="1.0" encoding="utf-8" ?>
|
||||||
|
<odoo>
|
||||||
|
<record id="action_plan_print_tree" model="ir.ui.view">
|
||||||
|
<field name="name">sf.demand.plan.print.wizard.tree</field>
|
||||||
|
<field name="model">sf.demand.plan.print.wizard</field>
|
||||||
|
<field name="arch" type="xml">
|
||||||
|
<tree string="打印" class="print_demand" js_class="print_demand" >
|
||||||
|
<field name="model_id"/>
|
||||||
|
<field name="filename_url"/>
|
||||||
|
<field name="type"/>
|
||||||
|
<field name="machining_drawings" attrs="{'column_invisible': True }"/>
|
||||||
|
<field name="cnc_worksheet" attrs="{'column_invisible': True }" />
|
||||||
|
<field name="status"/>
|
||||||
|
</tree>
|
||||||
|
</field>
|
||||||
|
</record>
|
||||||
|
</odoo>
|
||||||
@@ -3,6 +3,7 @@ import logging
|
|||||||
|
|
||||||
import re
|
import re
|
||||||
from odoo import models, fields, api
|
from odoo import models, fields, api
|
||||||
|
from odoo.exceptions import ValidationError
|
||||||
|
|
||||||
|
|
||||||
class ResProductCategory(models.Model):
|
class ResProductCategory(models.Model):
|
||||||
@@ -17,7 +18,7 @@ class ResProductCategory(models.Model):
|
|||||||
class ResProductProduct(models.Model):
|
class ResProductProduct(models.Model):
|
||||||
_inherit = 'product.product'
|
_inherit = 'product.product'
|
||||||
|
|
||||||
single_manufacturing = fields.Boolean(string="单个制造")
|
# single_manufacturing = fields.Boolean(string="单个制造")
|
||||||
is_bfm = fields.Boolean('业务平台是否自动创建', default=False)
|
is_bfm = fields.Boolean('业务平台是否自动创建', default=False)
|
||||||
|
|
||||||
|
|
||||||
@@ -47,11 +48,14 @@ class ResMrpBomMo(models.Model):
|
|||||||
item.subcontractor_name = ''
|
item.subcontractor_name = ''
|
||||||
|
|
||||||
def bom_create_line_has(self, embryo):
|
def bom_create_line_has(self, embryo):
|
||||||
|
product = self.product_tmpl_id
|
||||||
|
if product.unit_number in (0, None, False):
|
||||||
|
raise ValidationError(f'产品{product.name}单件用量的值不能为{product.unit_number}')
|
||||||
vals = {
|
vals = {
|
||||||
'bom_id': self.id,
|
'bom_id': self.id,
|
||||||
'product_id': embryo.id,
|
'product_id': embryo.id,
|
||||||
'product_tmpl_id': embryo.product_tmpl_id.id,
|
'product_tmpl_id': embryo.product_tmpl_id.id,
|
||||||
'product_qty': 1,
|
'product_qty': product.unit_number,
|
||||||
'product_uom_id': 1
|
'product_uom_id': 1
|
||||||
}
|
}
|
||||||
return self.env['mrp.bom.line'].sudo().create(vals)
|
return self.env['mrp.bom.line'].sudo().create(vals)
|
||||||
@@ -122,7 +126,7 @@ class ResMrpBomMo(models.Model):
|
|||||||
# 查bom的原材料
|
# 查bom的原材料
|
||||||
def get_raw_bom(self, product):
|
def get_raw_bom(self, product):
|
||||||
raw_bom = self.env['product.product'].search(
|
raw_bom = self.env['product.product'].search(
|
||||||
[('categ_id.type', '=', '原材料'), ('materials_type_id', '=', product.materials_type_id.id)],limit=1)
|
[('categ_id.type', '=', '原材料'), ('materials_type_id', '=', product.materials_type_id.id)], limit=1)
|
||||||
return raw_bom
|
return raw_bom
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -10,6 +10,7 @@ class ResProductTemplate(models.Model):
|
|||||||
model_name = fields.Char('模型名称')
|
model_name = fields.Char('模型名称')
|
||||||
categ_type = fields.Selection(
|
categ_type = fields.Selection(
|
||||||
[("成品", "成品"), ("胚料", "胚料"), ("原材料", "原材料")], string='产品的类别', related='categ_id.type', store=True)
|
[("成品", "成品"), ("胚料", "胚料"), ("原材料", "原材料")], string='产品的类别', related='categ_id.type', store=True)
|
||||||
|
blank_type = fields.Selection([('圆料', '圆料'), ('方料', '方料')], string='坯料分类')
|
||||||
model_long = fields.Float('模型长[mm]', digits=(16, 3))
|
model_long = fields.Float('模型长[mm]', digits=(16, 3))
|
||||||
model_width = fields.Float('模型宽[mm]', digits=(16, 3))
|
model_width = fields.Float('模型宽[mm]', digits=(16, 3))
|
||||||
model_height = fields.Float('模型高[mm]', digits=(16, 3))
|
model_height = fields.Float('模型高[mm]', digits=(16, 3))
|
||||||
@@ -72,14 +73,20 @@ class ResProductTemplate(models.Model):
|
|||||||
copy_product_id.product_tmpl_id.active = True
|
copy_product_id.product_tmpl_id.active = True
|
||||||
model_type = self.env['sf.model.type'].search([], limit=1)
|
model_type = self.env['sf.model.type'].search([], limit=1)
|
||||||
attachment = self.attachment_create(item['model_name'], item['model_data'])
|
attachment = self.attachment_create(item['model_name'], item['model_data'])
|
||||||
|
# 判断参数中是否包含 坯料尺寸(长、宽、高)
|
||||||
|
blank_bool = any(value is not None and value != 0 for value in (
|
||||||
|
item.get('blank_length'), item.get('blank_width'), item.get('blank_height'))) if all(
|
||||||
|
key in item for key in ('blank_length', 'blank_width', 'blank_height')) else False
|
||||||
vals = {
|
vals = {
|
||||||
'name': '%s-%s-%s' % ('P', order_id.name, i),
|
'name': '%s-%s-%s' % ('P', order_id.name, i),
|
||||||
'model_long': item['model_long'] + model_type.embryo_tolerance,
|
'blank_type': item.get('blank_type'),
|
||||||
'model_width': item['model_width'] + model_type.embryo_tolerance,
|
'model_long': item.get('blank_length') if blank_bool else item['model_long'] + model_type.embryo_tolerance,
|
||||||
'model_height': item['model_height'] + model_type.embryo_tolerance,
|
'model_width': item.get('blank_width') if blank_bool else item['model_width'] + model_type.embryo_tolerance,
|
||||||
'model_volume': (item['model_long'] + model_type.embryo_tolerance) * (
|
'model_height': item.get('blank_height') if blank_bool else item['model_height'] + model_type.embryo_tolerance,
|
||||||
item['model_width'] + model_type.embryo_tolerance) * (
|
'model_volume': ((item['model_long'] + model_type.embryo_tolerance) *
|
||||||
item['model_height'] + model_type.embryo_tolerance),
|
(item['model_width'] + model_type.embryo_tolerance) *
|
||||||
|
(item['model_height'] + model_type.embryo_tolerance)) if not blank_bool else (
|
||||||
|
item.get('blank_length') * item.get('blank_width') * item.get('blank_height')),
|
||||||
'product_model_type_id': model_type.id,
|
'product_model_type_id': model_type.id,
|
||||||
'model_processing_panel': 'R',
|
'model_processing_panel': 'R',
|
||||||
'model_machining_precision': item['model_machining_precision'],
|
'model_machining_precision': item['model_machining_precision'],
|
||||||
|
|||||||
@@ -1,16 +1,16 @@
|
|||||||
import logging
|
# import logging
|
||||||
from odoo import fields, models, api
|
# from odoo import fields, models, api
|
||||||
from odoo.exceptions import UserError
|
# from odoo.exceptions import UserError
|
||||||
from odoo.tools import str2bool
|
# from odoo.tools import str2bool
|
||||||
|
|
||||||
|
|
||||||
class ResMrpRoutingWorkcenter(models.Model):
|
# class ResMrpRoutingWorkcenter(models.Model):
|
||||||
_inherit = 'mrp.routing.workcenter'
|
# _inherit = 'mrp.routing.workcenter'
|
||||||
def init(self):
|
# def init(self):
|
||||||
super(ResMrpRoutingWorkcenter, self).init()
|
# super(ResMrpRoutingWorkcenter, self).init()
|
||||||
# 在模块初始化时触发计算字段的更新
|
# # 在模块初始化时触发计算字段的更新
|
||||||
records = self.search([])
|
# records = self.search([])
|
||||||
if str2bool(self.env['ir.config_parameter'].get_param('sf.production.process.parameter.is_init_workcenter',default='False')):
|
# if str2bool(self.env['ir.config_parameter'].get_param('sf.production.process.parameter.is_init_workcenter',default='False')):
|
||||||
return
|
# return
|
||||||
records.optional_process_parameters_date()
|
# records.optional_process_parameters_date()
|
||||||
self.env['ir.config_parameter'].set_param('sf.production.process.parameter.is_init_workcenter', True)
|
# self.env['ir.config_parameter'].set_param('sf.production.process.parameter.is_init_workcenter', True)
|
||||||
@@ -1,85 +1,85 @@
|
|||||||
# -*- coding: utf-8 -*-
|
# # -*- coding: utf-8 -*-
|
||||||
import logging
|
# import logging
|
||||||
from odoo import fields, models, api
|
# from odoo import fields, models, api
|
||||||
from odoo.exceptions import UserError
|
# from odoo.exceptions import UserError
|
||||||
from odoo.tools import str2bool
|
# from odoo.tools import str2bool
|
||||||
|
|
||||||
|
|
||||||
class SfProductionProcessParameter(models.Model):
|
# class SfProductionProcessParameter(models.Model):
|
||||||
_inherit = 'sf.production.process.parameter'
|
# _inherit = 'sf.production.process.parameter'
|
||||||
|
|
||||||
|
|
||||||
@api.model
|
# @api.model
|
||||||
def create(self, vals):
|
# def create(self, vals):
|
||||||
# if vals.get('code', '/') == '/' or vals.get('code', '/') is False:
|
# # if vals.get('code', '/') == '/' or vals.get('code', '/') is False:
|
||||||
# vals['code'] = '101'+self.routing_id.code +self.env['ir.sequence'].next_by_code('sf.production.process.parameter')
|
# # vals['code'] = '101'+self.routing_id.code +self.env['ir.sequence'].next_by_code('sf.production.process.parameter')
|
||||||
if vals.get('routing_id'):
|
# if vals.get('routing_id'):
|
||||||
# vals['gain_way'] = '外协'
|
# # vals['gain_way'] = '外协'
|
||||||
routing_id = self.env['mrp.routing.workcenter'].browse(vals.get('routing_id'))
|
# routing_id = self.env['mrp.routing.workcenter'].browse(vals.get('routing_id'))
|
||||||
if routing_id.surface_technics_id and not vals.get('process_id'):
|
# if routing_id.surface_technics_id and not vals.get('process_id'):
|
||||||
vals['process_id'] = routing_id.surface_technics_id.id
|
# vals['process_id'] = routing_id.surface_technics_id.id
|
||||||
if vals.get('code', '/') == '/' or vals.get('code', '/') is False:
|
# if vals.get('code', '/') == '/' or vals.get('code', '/') is False:
|
||||||
vals['code'] = '101' + routing_id.code + self.env['ir.sequence'].next_by_code(
|
# vals['code'] = '101' + routing_id.code + self.env['ir.sequence'].next_by_code(
|
||||||
'sf.production.process.parameter')
|
# 'sf.production.process.parameter')
|
||||||
obj = super(SfProductionProcessParameter, self).create(vals)
|
# obj = super(SfProductionProcessParameter, self).create(vals)
|
||||||
return obj
|
# return obj
|
||||||
def create_service_product(self):
|
# def create_service_product(self):
|
||||||
service_categ = self.env.ref(
|
# service_categ = self.env.ref(
|
||||||
'sf_dlm.product_category_surface_technics_sf').sudo()
|
# 'sf_dlm.product_category_surface_technics_sf').sudo()
|
||||||
|
|
||||||
product_name = f"{self.process_id.name}_{self.name}"
|
# product_name = f"{self.process_id.name}_{self.name}"
|
||||||
product_id = self.env['product.template'].search(
|
# product_id = self.env['product.template'].search(
|
||||||
[("name", '=', product_name)])
|
# [("name", '=', product_name)])
|
||||||
if product_id:
|
# if product_id:
|
||||||
product_id.server_product_process_parameters_id = self.id
|
# product_id.server_product_process_parameters_id = self.id
|
||||||
else:
|
# else:
|
||||||
res_partner = self.env['res.partner'].search([('name','=','湖南傲派自动化设备有限公司')])
|
# res_partner = self.env['res.partner'].search([('name','=','湖南傲派自动化设备有限公司')])
|
||||||
self.env['product.template'].create({
|
# self.env['product.template'].create({
|
||||||
'detailed_type': 'service',
|
# 'detailed_type': 'service',
|
||||||
'name': product_name,
|
# 'name': product_name,
|
||||||
'invoice_policy': 'delivery',
|
# 'invoice_policy': 'delivery',
|
||||||
'categ_id': service_categ.id,
|
# 'categ_id': service_categ.id,
|
||||||
'description': f"基于{self.name}创建的服务产品",
|
# 'description': f"基于{self.name}创建的服务产品",
|
||||||
'sale_ok': True, # 可销售
|
# 'sale_ok': True, # 可销售
|
||||||
'purchase_ok': True, # 可采购
|
# 'purchase_ok': True, # 可采购
|
||||||
'server_product_process_parameters_id': self.id,
|
# 'server_product_process_parameters_id': self.id,
|
||||||
'seller_ids': [(0, 0, {
|
# 'seller_ids': [(0, 0, {
|
||||||
# 'delay': 1,
|
# # 'delay': 1,
|
||||||
'partner_id': res_partner.id,
|
# 'partner_id': res_partner.id,
|
||||||
'price': 1, })],
|
# 'price': 1, })],
|
||||||
})
|
# })
|
||||||
|
|
||||||
def create_work_center(self):
|
# def create_work_center(self):
|
||||||
production_process_parameter = self
|
# production_process_parameter = self
|
||||||
if not production_process_parameter.process_id:
|
# if not production_process_parameter.process_id:
|
||||||
return
|
# return
|
||||||
if not production_process_parameter.routing_id:
|
# if not production_process_parameter.routing_id:
|
||||||
workcenter_id = self.env['mrp.routing.workcenter'].search(
|
# workcenter_id = self.env['mrp.routing.workcenter'].search(
|
||||||
[("surface_technics_id", '=', production_process_parameter.process_id.id)])
|
# [("surface_technics_id", '=', production_process_parameter.process_id.id)])
|
||||||
if not workcenter_id:
|
# if not workcenter_id:
|
||||||
outsourcing_work_center = self.env['mrp.workcenter'].search(
|
# outsourcing_work_center = self.env['mrp.workcenter'].search(
|
||||||
[("name", '=', '外协工作中心')])
|
# [("name", '=', '外协工作中心')])
|
||||||
routing_id = self.env['mrp.routing.workcenter'].create({
|
# routing_id = self.env['mrp.routing.workcenter'].create({
|
||||||
'workcenter_ids': [(6, 0, outsourcing_work_center.ids)],
|
# 'workcenter_ids': [(6, 0, outsourcing_work_center.ids)],
|
||||||
'routing_tag': 'special',
|
# 'routing_tag': 'special',
|
||||||
'routing_type': '表面工艺',
|
# 'routing_type': '表面工艺',
|
||||||
'is_outsource': True,
|
# 'is_outsource': True,
|
||||||
'surface_technics_id': production_process_parameter.process_id.id,
|
# 'surface_technics_id': production_process_parameter.process_id.id,
|
||||||
'name': production_process_parameter.process_id.name,
|
# 'name': production_process_parameter.process_id.name,
|
||||||
})
|
# })
|
||||||
production_process_parameter.routing_id = routing_id.id
|
# production_process_parameter.routing_id = routing_id.id
|
||||||
else:
|
# else:
|
||||||
production_process_parameter.routing_id = workcenter_id.id
|
# production_process_parameter.routing_id = workcenter_id.id
|
||||||
|
|
||||||
def init(self):
|
# def init(self):
|
||||||
super(SfProductionProcessParameter, self).init()
|
# super(SfProductionProcessParameter, self).init()
|
||||||
# 在模块初始化时触发计算字段的更新
|
# # 在模块初始化时触发计算字段的更新
|
||||||
records = self.search([])
|
# records = self.search([])
|
||||||
if str2bool(self.env['ir.config_parameter'].get_param('sf.production.process.parameter.is_init_process',
|
# if str2bool(self.env['ir.config_parameter'].get_param('sf.production.process.parameter.is_init_process',
|
||||||
default='False')):
|
# default='False')):
|
||||||
return
|
# return
|
||||||
for record in records:
|
# for record in records:
|
||||||
if not record.outsourced_service_products:
|
# if not record.outsourced_service_products:
|
||||||
record.create_service_product()
|
# record.create_service_product()
|
||||||
record.create_work_center()
|
# record.create_work_center()
|
||||||
self.env['ir.config_parameter'].set_param('sf.production.process.parameter.is_init_process', True)
|
# self.env['ir.config_parameter'].set_param('sf.production.process.parameter.is_init_process', True)
|
||||||
|
|||||||
@@ -95,7 +95,9 @@
|
|||||||
<page string="加工参数">
|
<page string="加工参数">
|
||||||
<group>
|
<group>
|
||||||
<group string="模型">
|
<group string="模型">
|
||||||
<label for="model_long" string="尺寸[mm]"/>
|
<field name="blank_type" readonly="1"/>
|
||||||
|
<field name="blank_precision" readonly="1"/>
|
||||||
|
<label for="model_long" string="坯料尺寸[mm]"/>
|
||||||
<div class="o_address_format">
|
<div class="o_address_format">
|
||||||
<label for="model_long" string="长"/>
|
<label for="model_long" string="长"/>
|
||||||
<field name="model_long" class="o_address_zip"/>
|
<field name="model_long" class="o_address_zip"/>
|
||||||
@@ -104,6 +106,7 @@
|
|||||||
<label for="model_height" string="高"/>
|
<label for="model_height" string="高"/>
|
||||||
<field name="model_height" class="o_address_zip"/>
|
<field name="model_height" class="o_address_zip"/>
|
||||||
</div>
|
</div>
|
||||||
|
<field name="unit_number" readonly="1"/>
|
||||||
<field name="model_volume" string="体积[mm³]"/>
|
<field name="model_volume" string="体积[mm³]"/>
|
||||||
<field name="product_model_type_id" string="模型类型"/>
|
<field name="product_model_type_id" string="模型类型"/>
|
||||||
<field name="model_processing_panel" placeholder="例如R,U" string="加工面板"
|
<field name="model_processing_panel" placeholder="例如R,U" string="加工面板"
|
||||||
|
|||||||
@@ -377,7 +377,11 @@ class Sf_Dashboard_Connect(http.Controller):
|
|||||||
line_list_obj = request.env['sf.production.line'].sudo().search([('name', 'ilike', 'CNC')])
|
line_list_obj = request.env['sf.production.line'].sudo().search([('name', 'ilike', 'CNC')])
|
||||||
line_list = list(map(lambda x: x.name, line_list_obj))
|
line_list = list(map(lambda x: x.name, line_list_obj))
|
||||||
# print('line_list: %s' % line_list)
|
# print('line_list: %s' % line_list)
|
||||||
res['LineList'] = line_list
|
res['LineList'] = ['业绩总览']
|
||||||
|
res['LineList'] += line_list
|
||||||
|
res['LineList'].append('人工线下加工中心')
|
||||||
|
# 增加“业绩总览”与“人工线下加工中心”
|
||||||
|
|
||||||
|
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
res = {'Succeed': False, 'ErrorCode': 202, 'Error': e}
|
res = {'Succeed': False, 'ErrorCode': 202, 'Error': e}
|
||||||
@@ -401,37 +405,55 @@ class Sf_Dashboard_Connect(http.Controller):
|
|||||||
|
|
||||||
try:
|
try:
|
||||||
plan_obj = request.env['sf.production.plan'].sudo()
|
plan_obj = request.env['sf.production.plan'].sudo()
|
||||||
production_obj = request.env['mrp.production'].sudo()
|
# production_obj = request.env['mrp.production'].sudo()
|
||||||
work_order_obj = request.env['mrp.workorder'].sudo()
|
work_order_obj = request.env['mrp.workorder'].sudo()
|
||||||
line_list = ast.literal_eval(kw['line_list'])
|
line_list = ast.literal_eval(kw['line_list'])
|
||||||
|
|
||||||
|
line_list_obj = request.env['sf.production.line'].sudo().search([('name', 'ilike', 'CNC')])
|
||||||
|
cnc_line_list = list(map(lambda x: x.name, line_list_obj))
|
||||||
# print('line_list: %s' % line_list)
|
# print('line_list: %s' % line_list)
|
||||||
for line in line_list:
|
for line in line_list:
|
||||||
|
|
||||||
|
if line == '业绩总览':
|
||||||
|
work_order_domain = [('routing_type', 'in', ['人工线下加工', 'CNC加工'])]
|
||||||
|
plan_domain = []
|
||||||
|
elif line == '人工线下加工中心':
|
||||||
|
work_order_domain = [('routing_type', '=', '人工线下加工')]
|
||||||
|
plan_domain = [('production_type', '=', '人工线下加工')]
|
||||||
|
else:
|
||||||
|
work_order_domain = [
|
||||||
|
('production_line_id.name', '=', line),
|
||||||
|
('routing_type', '=', 'CNC加工')
|
||||||
|
]
|
||||||
|
plan_domain = [('production_line_id.name', '=', line)]
|
||||||
# # 工单计划量
|
# # 工单计划量
|
||||||
# plan_data_total_counts = production_obj.search_count(
|
# plan_data_total_counts = production_obj.search_count(
|
||||||
# [('production_line_id.name', '=', line), ('state', 'not in', ['cancel']),
|
# [('production_line_id.name', '=', line), ('state', 'not in', ['cancel']),
|
||||||
# ('active', '=', True)])
|
# ('active', '=', True)])
|
||||||
|
|
||||||
# 工单计划量切换为CNC工单
|
# 工单计划量切换为CNC工单
|
||||||
plan_data_total_counts = work_order_obj.search_count(
|
plan_data_total = work_order_obj.search(work_order_domain + [
|
||||||
[('production_line_id.name', '=', line), ('id', '!=', 8061),
|
('id', '!=', 8061),
|
||||||
('state', 'in', ['ready', 'progress', 'done']), ('routing_type', '=', 'CNC加工')])
|
('state', 'in', ['ready', 'progress', 'done'])
|
||||||
|
])
|
||||||
|
|
||||||
|
plan_data_total_counts = sum(plan_data_total.mapped('qty_produced'))
|
||||||
|
|
||||||
# # 工单完成量
|
# # 工单完成量
|
||||||
# plan_data_finish_counts = plan_obj.search_count(
|
# plan_data_finish_counts = plan_obj.search_count(
|
||||||
# [('production_line_id.name', '=', line), ('state', 'in', ['finished'])])
|
# [('production_line_id.name', '=', line), ('state', 'in', ['finished'])])
|
||||||
|
|
||||||
# 工单完成量切换为CNC工单
|
# 工单完成量切换为CNC工单
|
||||||
plan_data_finish_counts = work_order_obj.search_count(
|
plan_data_finish = work_order_obj.search(work_order_domain + [
|
||||||
[('production_line_id.name', '=', line),
|
('state', 'in', ['done'])
|
||||||
('state', 'in', ['done']), ('routing_type', '=', 'CNC加工')])
|
])
|
||||||
|
|
||||||
|
plan_data_finish_counts = sum(plan_data_finish.mapped('qty_produced'))
|
||||||
|
|
||||||
# 超期完成量
|
# 超期完成量
|
||||||
# 搜索所有已经完成的工单
|
# 搜索所有已经完成的工单
|
||||||
plan_data_overtime = work_order_obj.search([
|
plan_data_overtime = work_order_obj.search(work_order_domain + [
|
||||||
('production_line_id.name', '=', line),
|
('state', 'in', ['done'])
|
||||||
('state', 'in', ['done']),
|
|
||||||
('routing_type', '=', 'CNC加工')
|
|
||||||
])
|
])
|
||||||
|
|
||||||
# 使用 filtered 进行字段比较
|
# 使用 filtered 进行字段比较
|
||||||
@@ -440,36 +462,38 @@ class Sf_Dashboard_Connect(http.Controller):
|
|||||||
)
|
)
|
||||||
|
|
||||||
# 获取数量
|
# 获取数量
|
||||||
plan_data_overtime_counts = len(plan_data_overtime_counts)
|
# plan_data_overtime_counts = len(plan_data_overtime_counts)
|
||||||
|
plan_data_overtime_counts = sum(plan_data_overtime_counts.mapped('qty_produced'))
|
||||||
|
|
||||||
# 查找符合条件的生产计划记录
|
# 查找符合条件的生产计划记录
|
||||||
plan_data = plan_obj.search([
|
# plan_data = plan_obj.search(plan_domain)
|
||||||
('production_line_id.name', '=', line),
|
|
||||||
])
|
|
||||||
|
|
||||||
# 过滤出那些检测结果状态为 '返工' 或 '报废' 的记录
|
# 过滤出那些检测结果状态为 '返工' 或 '报废' 的记录
|
||||||
# faulty_plans = plan_data.filtered(lambda p: any(
|
# faulty_plans = plan_data.filtered(lambda p: any(
|
||||||
# result.test_results in ['返工', '报废'] for result in p.production_id.detection_result_ids
|
# result.test_results in ['返工', '报废'] for result in p.production_id.detection_result_ids
|
||||||
# ))
|
# ))
|
||||||
|
|
||||||
faulty_plans = request.env['quality.check'].sudo().search([
|
faulty_plans = work_order_obj.search(work_order_domain + [
|
||||||
('operation_id.name', '=', 'CNC加工'),
|
('state', 'in', ['scrap', 'rework'])
|
||||||
('quality_state', 'in', ['fail'])
|
|
||||||
])
|
])
|
||||||
|
|
||||||
# 查找制造订单取消与归档的数量
|
# 查找制造订单取消与归档的数量
|
||||||
cancel_order_count = production_obj.search_count(
|
# cancel_order_count = production_obj.search_count(
|
||||||
[('production_line_id.name', '=', line), ('state', 'in', ['cancel']),
|
# [('production_line_id.name', '=', line), ('state', 'in', ['cancel']),
|
||||||
('active', '=', False)])
|
# ('active', '=', False)])
|
||||||
|
|
||||||
# 计算符合条件的记录数量
|
# 计算符合条件的记录数量
|
||||||
# plan_data_fault_counts = len(faulty_plans) + cancel_order_count
|
# plan_data_fault_counts = len(faulty_plans) + cancel_order_count
|
||||||
plan_data_fault_counts = len(faulty_plans)
|
# plan_data_fault_counts = len(faulty_plans)
|
||||||
|
plan_data_fault_counts = sum(faulty_plans.mapped('qty_produced'))
|
||||||
|
|
||||||
# 工单返工数量
|
# 工单返工数量
|
||||||
|
|
||||||
plan_data_rework_counts = plan_obj.search_count(
|
plan_data_rework = work_order_obj.search(work_order_domain + [
|
||||||
[('production_line_id.name', '=', line), ('production_id.state', 'in', ['rework'])])
|
('state', 'in', ['rework'])
|
||||||
|
])
|
||||||
|
|
||||||
|
plan_data_rework_counts = sum(plan_data_rework.mapped('qty_produced'))
|
||||||
|
|
||||||
# 工单完成率
|
# 工单完成率
|
||||||
finishe_rate = round(
|
finishe_rate = round(
|
||||||
@@ -479,8 +503,9 @@ class Sf_Dashboard_Connect(http.Controller):
|
|||||||
plan_data_progress_deviation = plan_data_total_counts - plan_data_finish_counts - plan_data_fault_counts
|
plan_data_progress_deviation = plan_data_total_counts - plan_data_finish_counts - plan_data_fault_counts
|
||||||
|
|
||||||
# 完成记录
|
# 完成记录
|
||||||
plan_data_finish_orders = plan_obj.search(
|
plan_data_finish_orders = plan_obj.search(plan_domain + [
|
||||||
[('production_line_id.name', '=', line), ('state', 'in', ['finished'])])
|
('state', 'in', ['finished'])
|
||||||
|
])
|
||||||
|
|
||||||
# # 检测量
|
# # 检测量
|
||||||
# detection_nums = 0
|
# detection_nums = 0
|
||||||
@@ -534,25 +559,25 @@ class Sf_Dashboard_Connect(http.Controller):
|
|||||||
delay_rate = round((delay_num / plan_data_finish_counts if plan_data_finish_counts > 0 else 0), 3)
|
delay_rate = round((delay_num / plan_data_finish_counts if plan_data_finish_counts > 0 else 0), 3)
|
||||||
on_time_rate = 1 - delay_rate
|
on_time_rate = 1 - delay_rate
|
||||||
|
|
||||||
if plan_data:
|
# if plan_data:
|
||||||
data = {
|
data = {
|
||||||
'plan_data_total_counts': plan_data_total_counts,
|
'plan_data_total_counts': plan_data_total_counts,
|
||||||
'plan_data_finish_counts': plan_data_finish_counts,
|
'plan_data_finish_counts': plan_data_finish_counts,
|
||||||
'plan_data_plan_counts': plan_data_total_counts,
|
'plan_data_plan_counts': plan_data_total_counts,
|
||||||
'plan_data_fault_counts': plan_data_fault_counts,
|
'plan_data_fault_counts': plan_data_fault_counts,
|
||||||
'nopass_orders_counts': detection_data - len(pass_nums),
|
'nopass_orders_counts': detection_data - len(pass_nums),
|
||||||
'finishe_rate': finishe_rate,
|
'finishe_rate': finishe_rate,
|
||||||
'plan_data_progress_deviation': plan_data_progress_deviation,
|
'plan_data_progress_deviation': plan_data_progress_deviation,
|
||||||
'plan_data_rework_counts': plan_data_rework_counts,
|
'plan_data_rework_counts': plan_data_rework_counts,
|
||||||
'on_time_rate': on_time_rate,
|
'on_time_rate': on_time_rate,
|
||||||
# 'detection_data': detection_data,
|
# 'detection_data': detection_data,
|
||||||
'detection_data': plan_data_finish_counts,
|
'detection_data': plan_data_finish_counts,
|
||||||
'pass_rate': (plan_data_finish_counts - plan_data_fault_counts) / plan_data_finish_counts,
|
'pass_rate': (plan_data_finish_counts - plan_data_fault_counts) / plan_data_finish_counts,
|
||||||
'plan_data_overtime_counts': plan_data_overtime_counts,
|
'plan_data_overtime_counts': plan_data_overtime_counts,
|
||||||
'overtime_rate': plan_data_overtime_counts / plan_data_finish_counts
|
'overtime_rate': plan_data_overtime_counts / plan_data_finish_counts
|
||||||
if plan_data_finish_counts > 0 else 0,
|
if plan_data_finish_counts > 0 else 0,
|
||||||
}
|
}
|
||||||
res['data'][line] = data
|
res['data'][line] = data
|
||||||
|
|
||||||
return json.dumps(res) # 注意使用 json.dumps 而不是直接用 json.JSONEncoder().encode()
|
return json.dumps(res) # 注意使用 json.dumps 而不是直接用 json.JSONEncoder().encode()
|
||||||
|
|
||||||
@@ -608,16 +633,34 @@ class Sf_Dashboard_Connect(http.Controller):
|
|||||||
date_list.append(current_date)
|
date_list.append(current_date)
|
||||||
current_date += timedelta(days=1)
|
current_date += timedelta(days=1)
|
||||||
return date_list
|
return date_list
|
||||||
|
|
||||||
|
|
||||||
for line in line_list:
|
if time_unit == 'hour':
|
||||||
date_field_name = 'date_finished' # 替换为你模型中的实际字段名
|
|
||||||
order_counts = []
|
for line in line_list:
|
||||||
|
date_field_name = 'date_finished' # 替换为你模型中的实际字段名
|
||||||
|
order_counts = []
|
||||||
|
|
||||||
if time_unit == 'hour':
|
if line == '业绩总览':
|
||||||
|
work_order_domain = [('routing_type', 'in', ['人工线下加工', 'CNC加工'])]
|
||||||
|
elif line == '人工线下加工中心':
|
||||||
|
work_order_domain = [('routing_type', '=', '人工线下加工')]
|
||||||
|
else:
|
||||||
|
work_order_domain = [
|
||||||
|
('production_line_id.name', '=', line),
|
||||||
|
('routing_type', '=', 'CNC加工')
|
||||||
|
]
|
||||||
time_intervals = get_time_intervals(begin_time, end_time, time_unit)
|
time_intervals = get_time_intervals(begin_time, end_time, time_unit)
|
||||||
print('============================= %s' % time_intervals)
|
print('============================= %s' % time_intervals)
|
||||||
|
|
||||||
time_count_dict = {}
|
time_count_dict = {}
|
||||||
|
plan_count_dict = {}
|
||||||
|
|
||||||
|
orders = request.env['mrp.workorder'].sudo().search(work_order_domain + [
|
||||||
|
('state', 'in', ['done']),
|
||||||
|
(date_field_name, '>=', begin_time.strftime('%Y-%m-%d %H:%M:%S')),
|
||||||
|
(date_field_name, '<=', end_time.strftime('%Y-%m-%d %H:%M:%S'))
|
||||||
|
])
|
||||||
|
|
||||||
for time_interval in time_intervals:
|
for time_interval in time_intervals:
|
||||||
start_time, end_time = time_interval
|
start_time, end_time = time_interval
|
||||||
@@ -629,66 +672,113 @@ class Sf_Dashboard_Connect(http.Controller):
|
|||||||
# (date_field_name, '<=', end_time.strftime('%Y-%m-%d %H:%M:%S')) # 包括结束时间
|
# (date_field_name, '<=', end_time.strftime('%Y-%m-%d %H:%M:%S')) # 包括结束时间
|
||||||
# ])
|
# ])
|
||||||
|
|
||||||
orders = request.env['mrp.workorder'].sudo().search([
|
interval_orders = orders.filtered(
|
||||||
('routing_type', '=', 'CNC加工'), # 将第一个条件合并进来
|
lambda o: o[date_field_name] >= start_time
|
||||||
('production_line_id.name', '=', line),
|
and o[date_field_name] <= end_time
|
||||||
('state', 'in', ['done']),
|
)
|
||||||
(date_field_name, '>=', start_time.strftime('%Y-%m-%d %H:%M:%S')),
|
|
||||||
(date_field_name, '<=', end_time.strftime('%Y-%m-%d %H:%M:%S'))
|
|
||||||
])
|
|
||||||
|
|
||||||
# 使用小时和分钟作为键,确保每个小时的数据有独立的键
|
# 使用小时和分钟作为键,确保每个小时的数据有独立的键
|
||||||
key = start_time.strftime('%H:%M:%S') # 只取小时:分钟:秒作为键
|
key = start_time.strftime('%H:%M:%S') # 只取小时:分钟:秒作为键
|
||||||
time_count_dict[key] = len(orders)
|
# time_count_dict[key] = len(orders)
|
||||||
|
time_count_dict[key] = sum(interval_orders.mapped('qty_produced'))
|
||||||
|
|
||||||
|
# 计划量,目前只能从mail.message中筛选出
|
||||||
|
plan_order_messages = request.env['mail.message'].sudo().search([
|
||||||
|
('model', '=', 'mrp.workorder'),
|
||||||
|
('create_date', '>=', begin_time.strftime('%Y-%m-%d %H:%M:%S')),
|
||||||
|
('create_date', '<=', end_time.strftime('%Y-%m-%d %H:%M:%S')),
|
||||||
|
('tracking_value_ids.field_desc', '=', '状态'),
|
||||||
|
('tracking_value_ids.new_value_char', '=', '就绪')
|
||||||
|
])
|
||||||
|
|
||||||
|
for time_interval in time_intervals:
|
||||||
|
start_time, end_time = time_interval
|
||||||
|
|
||||||
|
# orders = plan_obj.search([
|
||||||
|
# ('production_line_id.name', '=', line),
|
||||||
|
# ('state', 'in', ['done']),
|
||||||
|
# (date_field_name, '>=', start_time.strftime('%Y-%m-%d %H:%M:%S')),
|
||||||
|
# (date_field_name, '<=', end_time.strftime('%Y-%m-%d %H:%M:%S')) # 包括结束时间
|
||||||
|
# ])
|
||||||
|
|
||||||
|
interval_plan_orders = plan_order_messages.filtered(
|
||||||
|
lambda o: o.create_date >= start_time
|
||||||
|
and o.create_date <= end_time
|
||||||
|
)
|
||||||
|
|
||||||
|
interval_orders = request.env['mrp.workorder'].sudo().browse(interval_plan_orders.mapped('res_id'))
|
||||||
|
if line == '业绩总览':
|
||||||
|
interval_orders = interval_orders.filtered(lambda o: o.routing_type in ['人工线下加工', 'CNC加工'])
|
||||||
|
elif line == '人工线下加工中心':
|
||||||
|
interval_orders = interval_orders.filtered(lambda o: o.routing_type == '人工线下加工')
|
||||||
|
else:
|
||||||
|
interval_orders = interval_orders.filtered(lambda o: o.routing_type == 'CNC加工' and o.production_line_id.name == line)
|
||||||
|
|
||||||
|
# 使用小时和分钟作为键,确保每个小时的数据有独立的键
|
||||||
|
key = start_time.strftime('%H:%M:%S') # 只取小时:分钟:秒作为键
|
||||||
|
# time_count_dict[key] = len(orders)
|
||||||
|
plan_count_dict[key] = sum(interval_orders.mapped('qty_produced'))
|
||||||
|
|
||||||
# order_counts.append()
|
# order_counts.append()
|
||||||
res['data'][line] = {
|
res['data'][line] = {
|
||||||
'finish_order_nums': time_count_dict,
|
'finish_order_nums': time_count_dict,
|
||||||
'plan_order_nums': 28
|
'plan_order_nums': plan_count_dict
|
||||||
}
|
}
|
||||||
return json.dumps(res)
|
else:
|
||||||
|
|
||||||
date_list = get_date_list(begin_time, end_time)
|
for line in line_list:
|
||||||
|
date_field_name = 'date_finished' # 替换为你模型中的实际字段名
|
||||||
|
order_counts = []
|
||||||
|
|
||||||
for date in date_list:
|
if line == '业绩总览':
|
||||||
next_day = date + timedelta(days=1)
|
work_order_domain = [('routing_type', 'in', ['人工线下加工', 'CNC加工'])]
|
||||||
orders = request.env['mrp.workorder'].sudo().search(
|
elif line == '人工线下加工中心':
|
||||||
[('production_id.production_line_id.name', '=', line), ('state', 'in', ['done']),
|
work_order_domain = [('routing_type', '=', '人工线下加工')]
|
||||||
('routing_type', '=', 'CNC加工'),
|
else:
|
||||||
(date_field_name, '>=', date.strftime('%Y-%m-%d 00:00:00')),
|
work_order_domain = [
|
||||||
(date_field_name, '<', next_day.strftime('%Y-%m-%d 00:00:00'))
|
('production_line_id.name', '=', line),
|
||||||
])
|
('routing_type', '=', 'CNC加工')
|
||||||
|
]
|
||||||
|
|
||||||
rework_orders = request.env['mrp.workorder'].sudo().search(
|
date_list = get_date_list(begin_time, end_time)
|
||||||
[('production_id.production_line_id.name', '=', line), ('state', 'in', ['rework']),
|
|
||||||
('routing_type', '=', 'CNC加工'),
|
|
||||||
(date_field_name, '>=', date.strftime('%Y-%m-%d 00:00:00')),
|
|
||||||
(date_field_name, '<', next_day.strftime('%Y-%m-%d 00:00:00'))
|
|
||||||
])
|
|
||||||
not_passed_orders = request.env['mrp.workorder'].sudo().search(
|
|
||||||
[('production_id.production_line_id.name', '=', line), ('state', 'in', ['scrap', 'cancel']),
|
|
||||||
('routing_type', '=', 'CNC加工'),
|
|
||||||
(date_field_name, '>=', date.strftime('%Y-%m-%d 00:00:00')),
|
|
||||||
(date_field_name, '<', next_day.strftime('%Y-%m-%d 00:00:00'))
|
|
||||||
])
|
|
||||||
order_counts.append({
|
|
||||||
'date': date.strftime('%Y-%m-%d'),
|
|
||||||
'order_count': len(orders),
|
|
||||||
'rework_orders': len(rework_orders),
|
|
||||||
'not_passed_orders': len(not_passed_orders)
|
|
||||||
})
|
|
||||||
# 外面包一层,没什么是包一层不能解决的,包一层就能区分了,类似于包一层div
|
|
||||||
# 外面包一层的好处是,可以把多个数据结构打包在一起,方便前端处理
|
|
||||||
|
|
||||||
# date_list_dict = {line: order_counts}
|
for date in date_list:
|
||||||
|
next_day = date + timedelta(days=1)
|
||||||
|
orders = request.env['mrp.workorder'].sudo().search(work_order_domain + [
|
||||||
|
('state', 'in', ['done']),
|
||||||
|
(date_field_name, '>=', date.strftime('%Y-%m-%d 00:00:00')),
|
||||||
|
(date_field_name, '<', next_day.strftime('%Y-%m-%d 00:00:00'))
|
||||||
|
])
|
||||||
|
|
||||||
res['data'][line] = order_counts
|
rework_orders = request.env['mrp.workorder'].sudo().search(work_order_domain + [
|
||||||
|
('state', 'in', ['rework']),
|
||||||
|
(date_field_name, '>=', date.strftime('%Y-%m-%d 00:00:00')),
|
||||||
|
(date_field_name, '<', next_day.strftime('%Y-%m-%d 00:00:00'))
|
||||||
|
])
|
||||||
|
not_passed_orders = request.env['mrp.workorder'].sudo().search(work_order_domain + [
|
||||||
|
('state', 'in', ['scrap', 'cancel']),
|
||||||
|
(date_field_name, '>=', date.strftime('%Y-%m-%d 00:00:00')),
|
||||||
|
(date_field_name, '<', next_day.strftime('%Y-%m-%d 00:00:00'))
|
||||||
|
])
|
||||||
|
order_counts.append({
|
||||||
|
'date': date.strftime('%Y-%m-%d'),
|
||||||
|
'order_count': sum(orders.mapped('qty_produced')),
|
||||||
|
'rework_orders': sum(rework_orders.mapped('qty_produced')),
|
||||||
|
'not_passed_orders': sum(not_passed_orders.mapped('qty_produced'))
|
||||||
|
})
|
||||||
|
# 外面包一层,没什么是包一层不能解决的,包一层就能区分了,类似于包一层div
|
||||||
|
# 外面包一层的好处是,可以把多个数据结构打包在一起,方便前端处理
|
||||||
|
|
||||||
|
# date_list_dict = {line: order_counts}
|
||||||
|
|
||||||
|
res['data'][line] = order_counts
|
||||||
return json.dumps(res)
|
return json.dumps(res)
|
||||||
|
|
||||||
# 实时产量
|
# 实时产量
|
||||||
@http.route('/api/RealTimeProduct', type='http', auth='public', methods=['GET', 'POST'], csrf=False, cors="*")
|
@http.route('/api/RealTimeProduct', type='http', auth='public', methods=['GET', 'POST'], csrf=False, cors="*")
|
||||||
def RealTimeProduct(self, **kw):
|
def RealTimeProduct(self, **kw):
|
||||||
"""
|
"""
|
||||||
获取实时产量
|
获取实时产量(作废)
|
||||||
:param kw:
|
:param kw:
|
||||||
:return:
|
:return:
|
||||||
"""
|
"""
|
||||||
@@ -711,6 +801,21 @@ class Sf_Dashboard_Connect(http.Controller):
|
|||||||
|
|
||||||
# 当班计划量
|
# 当班计划量
|
||||||
for line in line_list:
|
for line in line_list:
|
||||||
|
|
||||||
|
if line == '业绩总览':
|
||||||
|
work_order_domain = [('routing_type', 'in', ['人工线下加工', 'CNC加工'])]
|
||||||
|
plan_domain = []
|
||||||
|
elif line == '人工线下加工中心':
|
||||||
|
work_order_domain = [('routing_type', '=', '人工线下加工')]
|
||||||
|
plan_domain = [('production_type', '=', '人工线下加工')]
|
||||||
|
else:
|
||||||
|
work_order_domain = [
|
||||||
|
('production_line_id.name', '=', line),
|
||||||
|
('routing_type', '=', 'CNC加工')
|
||||||
|
]
|
||||||
|
plan_domain = [('production_line_id.name', '=', line)]
|
||||||
|
|
||||||
|
|
||||||
plan_order_nums = plan_obj.search_count(
|
plan_order_nums = plan_obj.search_count(
|
||||||
[('production_line_id.name', '=', line), ('state', 'not in', ['draft']),
|
[('production_line_id.name', '=', line), ('state', 'not in', ['draft']),
|
||||||
('date_planned_start', '>=', begin_time),
|
('date_planned_start', '>=', begin_time),
|
||||||
@@ -752,10 +857,10 @@ class Sf_Dashboard_Connect(http.Controller):
|
|||||||
:param kw:
|
:param kw:
|
||||||
:return:
|
:return:
|
||||||
"""
|
"""
|
||||||
|
|
||||||
# res = {'status': 1, 'message': '成功', 'not_done_data': [], 'done_data': []}
|
# res = {'status': 1, 'message': '成功', 'not_done_data': [], 'done_data': []}
|
||||||
res = {'status': 1, 'message': '成功', 'data': {}}
|
res = {'status': 1, 'message': '成功', 'data': {}}
|
||||||
plan_obj = request.env['sf.production.plan'].sudo()
|
plan_obj = request.env['sf.production.plan'].sudo()
|
||||||
|
work_order_obj = request.env['mrp.workorder'].sudo()
|
||||||
line_list = ast.literal_eval(kw['line_list'])
|
line_list = ast.literal_eval(kw['line_list'])
|
||||||
begin_time_str = kw['begin_time'].strip('"')
|
begin_time_str = kw['begin_time'].strip('"')
|
||||||
end_time_str = kw['end_time'].strip('"')
|
end_time_str = kw['end_time'].strip('"')
|
||||||
@@ -765,28 +870,39 @@ class Sf_Dashboard_Connect(http.Controller):
|
|||||||
not_done_data = []
|
not_done_data = []
|
||||||
done_data = []
|
done_data = []
|
||||||
final_data = {}
|
final_data = {}
|
||||||
|
not_done_index = 1
|
||||||
|
done_index = 1
|
||||||
|
|
||||||
for line in line_list:
|
for line in line_list:
|
||||||
|
|
||||||
|
if line == '业绩总览':
|
||||||
|
work_order_domain = [('routing_type', 'in', ['人工线下加工', 'CNC加工'])]
|
||||||
|
elif line == '人工线下加工中心':
|
||||||
|
work_order_domain = [('routing_type', '=', '人工线下加工')]
|
||||||
|
else:
|
||||||
|
work_order_domain = [
|
||||||
|
('production_line_id.name', '=', line),
|
||||||
|
('routing_type', '=', 'CNC加工')
|
||||||
|
]
|
||||||
# 未完成订单
|
# 未完成订单
|
||||||
# not_done_orders = plan_obj.search(
|
# not_done_orders = plan_obj.search(
|
||||||
# [('production_line_id.name', '=', line), ('state', 'not in', ['finished']),
|
# [('production_line_id.name', '=', line), ('state', 'not in', ['finished']),
|
||||||
# ('production_id.state', 'not in', ['cancel', 'done']), ('active', '=', True)
|
# ('production_id.state', 'not in', ['cancel', 'done']), ('active', '=', True)
|
||||||
# ])
|
# ])
|
||||||
not_done_orders = request.env['mrp.workorder'].sudo().search(
|
not_done_orders = work_order_obj.search(work_order_domain +
|
||||||
[('production_line_id.name', '=', line), ('state', 'in', ['ready', 'progress']),
|
[('state', 'in', ['ready', 'progress'])], order='id asc'
|
||||||
('routing_type', '=', 'CNC加工')
|
)
|
||||||
])
|
|
||||||
|
|
||||||
# 完成订单
|
# 完成订单
|
||||||
# 获取当前时间,并计算24小时前的时间
|
# 获取当前时间,并计算24小时前的时间
|
||||||
current_time = datetime.now()
|
current_time = datetime.now()
|
||||||
time_24_hours_ago = current_time - timedelta(hours=24)
|
time_24_hours_ago = current_time - timedelta(hours=24)
|
||||||
|
|
||||||
finish_orders = plan_obj.search([
|
finish_orders = work_order_obj.search(work_order_domain + [
|
||||||
('production_line_id.name', '=', line), ('state', 'in', ['finished']),
|
('state', 'in', ['finished']),
|
||||||
('production_id.state', 'not in', ['cancel']), ('active', '=', True),
|
('production_id.state', 'not in', ['cancel']),
|
||||||
('actual_end_time', '>=', time_24_hours_ago)
|
('date_finished', '>=', time_24_hours_ago)
|
||||||
])
|
], order='id asc')
|
||||||
# print(finish_orders)
|
# print(finish_orders)
|
||||||
|
|
||||||
# 获取所有未完成订单的ID列表
|
# 获取所有未完成订单的ID列表
|
||||||
@@ -795,14 +911,14 @@ class Sf_Dashboard_Connect(http.Controller):
|
|||||||
finish_order_ids = [order.id for order in finish_orders]
|
finish_order_ids = [order.id for order in finish_orders]
|
||||||
|
|
||||||
# 对ID进行排序
|
# 对ID进行排序
|
||||||
sorted_order_ids = sorted(order_ids)
|
# sorted_order_ids = sorted(order_ids)
|
||||||
|
|
||||||
finish_sorted_order_ids = sorted(finish_order_ids)
|
# finish_sorted_order_ids = sorted(finish_order_ids)
|
||||||
|
|
||||||
# 创建ID与序号的对应关系
|
# 创建ID与序号的对应关系
|
||||||
id_to_sequence = {order_id: index + 1 for index, order_id in enumerate(sorted_order_ids)}
|
# id_to_sequence = {order_id: index + 1 for index, order_id in enumerate(sorted_order_ids)}
|
||||||
|
|
||||||
finish_id_to_sequence = {order_id: index + 1 for index, order_id in enumerate(finish_sorted_order_ids)}
|
# finish_id_to_sequence = {order_id: index + 1 for index, order_id in enumerate(finish_sorted_order_ids)}
|
||||||
|
|
||||||
# # 输出结果或进一步处理
|
# # 输出结果或进一步处理
|
||||||
# for order_id, sequence in id_to_sequence.items():
|
# for order_id, sequence in id_to_sequence.items():
|
||||||
@@ -833,16 +949,17 @@ class Sf_Dashboard_Connect(http.Controller):
|
|||||||
}
|
}
|
||||||
|
|
||||||
line_dict = {
|
line_dict = {
|
||||||
'sequence': id_to_sequence[order.id],
|
'sequence': not_done_index,
|
||||||
'workorder_name': order.production_id.name,
|
'workorder_name': order.production_id.name,
|
||||||
'blank_name': blank_name,
|
'blank_name': blank_name,
|
||||||
'material': material,
|
'material': material,
|
||||||
'dimensions': dimensions,
|
'dimensions': dimensions,
|
||||||
'order_qty': 1,
|
'order_qty': order.qty_production,
|
||||||
'state': state_dict[order.state],
|
'state': state_dict[order.state],
|
||||||
|
|
||||||
}
|
}
|
||||||
not_done_data.append(line_dict)
|
not_done_data.append(line_dict)
|
||||||
|
not_done_index += 1
|
||||||
|
|
||||||
for finish_order in finish_orders:
|
for finish_order in finish_orders:
|
||||||
if not finish_order.actual_end_time:
|
if not finish_order.actual_end_time:
|
||||||
@@ -861,17 +978,18 @@ class Sf_Dashboard_Connect(http.Controller):
|
|||||||
material = material_match.group(1) if material_match else 'No match found'
|
material = material_match.group(1) if material_match else 'No match found'
|
||||||
|
|
||||||
line_dict = {
|
line_dict = {
|
||||||
'sequence': finish_id_to_sequence[finish_order.id],
|
'sequence': done_index,
|
||||||
'workorder_name': finish_order.name,
|
'workorder_name': finish_order.name,
|
||||||
'blank_name': blank_name,
|
'blank_name': blank_name,
|
||||||
'material': material,
|
'material': material,
|
||||||
'dimensions': dimensions,
|
'dimensions': dimensions,
|
||||||
'order_qty': finish_order.product_qty,
|
'order_qty': order.qty_produced,
|
||||||
'finish_time': finish_order.actual_end_time.strftime(
|
'finish_time': finish_order.actual_end_time.strftime(
|
||||||
'%Y-%m-%d %H:%M:%S') if finish_order.actual_end_time else ' '
|
'%Y-%m-%d %H:%M:%S') if finish_order.actual_end_time else ' '
|
||||||
|
|
||||||
}
|
}
|
||||||
done_data.append(line_dict)
|
done_data.append(line_dict)
|
||||||
|
done_index += 1
|
||||||
|
|
||||||
# 开始包一层
|
# 开始包一层
|
||||||
res['data'][line] = {'not_done_data': not_done_data, 'done_data': done_data}
|
res['data'][line] = {'not_done_data': not_done_data, 'done_data': done_data}
|
||||||
@@ -1298,10 +1416,7 @@ class Sf_Dashboard_Connect(http.Controller):
|
|||||||
conn = psycopg2.connect(**db_config)
|
conn = psycopg2.connect(**db_config)
|
||||||
# 获取请求的机床数据
|
# 获取请求的机床数据
|
||||||
machine_list = ast.literal_eval(kw['machine_list'])
|
machine_list = ast.literal_eval(kw['machine_list'])
|
||||||
time_threshold = datetime.now() - timedelta(days=1)
|
time_threshold = datetime.now().replace(hour=0, minute=0, second=0, microsecond=0)
|
||||||
|
|
||||||
alarm_last_24_time = 0.0
|
|
||||||
alarm_all_time = 0.0
|
|
||||||
|
|
||||||
def fetch_result_as_dict(cursor):
|
def fetch_result_as_dict(cursor):
|
||||||
"""辅助函数:将查询结果转为字典"""
|
"""辅助函数:将查询结果转为字典"""
|
||||||
@@ -1311,6 +1426,9 @@ class Sf_Dashboard_Connect(http.Controller):
|
|||||||
# 获取当前时间的时间戳
|
# 获取当前时间的时间戳
|
||||||
current_timestamp = datetime.now().timestamp()
|
current_timestamp = datetime.now().timestamp()
|
||||||
for item in machine_list:
|
for item in machine_list:
|
||||||
|
alarm_last_24_time = 0.0
|
||||||
|
alarm_all_time = 0.0
|
||||||
|
|
||||||
euipment_obj = request.env['maintenance.equipment'].sudo().search([('code', '=', item)])
|
euipment_obj = request.env['maintenance.equipment'].sudo().search([('code', '=', item)])
|
||||||
# 机床上线时间段
|
# 机床上线时间段
|
||||||
first_online_duration = current_timestamp - euipment_obj.first_online_time.timestamp()
|
first_online_duration = current_timestamp - euipment_obj.first_online_time.timestamp()
|
||||||
@@ -1327,8 +1445,8 @@ class Sf_Dashboard_Connect(http.Controller):
|
|||||||
cur.execute("""
|
cur.execute("""
|
||||||
SELECT * FROM device_data
|
SELECT * FROM device_data
|
||||||
WHERE device_name = %s
|
WHERE device_name = %s
|
||||||
AND device_state in ('待机', '警告', '运行中') AND time >= %s AND process_time IS NOT NULL
|
AND device_state in ('待机', '警告', '运行中') AND time <= %s AND process_time IS NOT NULL
|
||||||
ORDER BY time ASC
|
ORDER BY time DESC
|
||||||
LIMIT 1;
|
LIMIT 1;
|
||||||
""", (item, time_threshold))
|
""", (item, time_threshold))
|
||||||
last_24_time = fetch_result_as_dict(cur)
|
last_24_time = fetch_result_as_dict(cur)
|
||||||
@@ -1348,12 +1466,13 @@ class Sf_Dashboard_Connect(http.Controller):
|
|||||||
alarm_last_24_nums = []
|
alarm_last_24_nums = []
|
||||||
with conn.cursor() as cur:
|
with conn.cursor() as cur:
|
||||||
cur.execute("""
|
cur.execute("""
|
||||||
SELECT DISTINCT ON (alarm_start_time) alarm_time, alarm_start_time
|
SELECT DISTINCT ON (alarm_start_time, alarm_time) alarm_time, alarm_start_time
|
||||||
FROM device_data
|
FROM device_data
|
||||||
WHERE device_name = %s
|
WHERE device_name = %s
|
||||||
AND alarm_start_time IS NOT NULL AND time >= %s;
|
AND alarm_start_time IS NOT NULL AND alarm_start_time::timestamp >= %s;
|
||||||
""", (item, time_threshold))
|
""", (item, time_threshold))
|
||||||
results = cur.fetchall()
|
results = cur.fetchall()
|
||||||
|
logging.info("results============:%s" % results)
|
||||||
for result in results:
|
for result in results:
|
||||||
alarm_last_24_nums.append(result[1])
|
alarm_last_24_nums.append(result[1])
|
||||||
if result[0]:
|
if result[0]:
|
||||||
|
|||||||
@@ -9,9 +9,13 @@ _logger = logging.getLogger(__name__)
|
|||||||
|
|
||||||
class FTP_P(FTP):
|
class FTP_P(FTP):
|
||||||
"""
|
"""
|
||||||
重写FTP类,重写dirs方法
|
重写FTP类,重写dirs方法,增加编码处理
|
||||||
"""
|
"""
|
||||||
|
def __init__(self, host='', user='', passwd='', acct='', timeout=None, encoding='gbk'):
|
||||||
|
"""初始化时指定编码方式"""
|
||||||
|
super().__init__(host, user, passwd, acct, timeout)
|
||||||
|
self.encoding = encoding
|
||||||
|
|
||||||
def dirs(self, *args):
|
def dirs(self, *args):
|
||||||
"""List a directory in long form.
|
"""List a directory in long form.
|
||||||
By default list current directory to stdout.
|
By default list current directory to stdout.
|
||||||
@@ -32,7 +36,50 @@ class FTP_P(FTP):
|
|||||||
tempdic['name'] = [file for file in r_files if file != "." and file != ".."]
|
tempdic['name'] = [file for file in r_files if file != "." and file != ".."]
|
||||||
# 去除. ..
|
# 去除. ..
|
||||||
return tempdic
|
return tempdic
|
||||||
# return [file for file in r_files if file != "." and file != ".."]
|
|
||||||
|
def nlst(self, *args):
|
||||||
|
"""Get a list of files in a directory."""
|
||||||
|
files = []
|
||||||
|
def append(line):
|
||||||
|
try:
|
||||||
|
if isinstance(line, bytes):
|
||||||
|
files.append(line.decode(self.encoding))
|
||||||
|
else:
|
||||||
|
files.append(line)
|
||||||
|
except UnicodeDecodeError:
|
||||||
|
files.append(line.decode('utf-8', errors='replace'))
|
||||||
|
cmd = 'NLST'
|
||||||
|
if args:
|
||||||
|
cmd = cmd + ' ' + args[0]
|
||||||
|
self.retrlines(cmd, append)
|
||||||
|
return files
|
||||||
|
|
||||||
|
def cwd(self, dirname):
|
||||||
|
"""Change to a directory."""
|
||||||
|
try:
|
||||||
|
if isinstance(dirname, bytes):
|
||||||
|
dirname = dirname.decode(self.encoding)
|
||||||
|
return super().cwd(dirname)
|
||||||
|
except UnicodeEncodeError:
|
||||||
|
return super().cwd(dirname.encode(self.encoding).decode('utf-8'))
|
||||||
|
|
||||||
|
def storbinary(self, cmd, fp, blocksize=8192, callback=None, rest=None):
|
||||||
|
"""Store a file in binary mode."""
|
||||||
|
try:
|
||||||
|
if isinstance(cmd, bytes):
|
||||||
|
cmd = cmd.decode(self.encoding)
|
||||||
|
return super().storbinary(cmd, fp, blocksize, callback, rest)
|
||||||
|
except UnicodeEncodeError:
|
||||||
|
return super().storbinary(cmd.encode(self.encoding).decode('utf-8'), fp, blocksize, callback, rest)
|
||||||
|
|
||||||
|
def retrbinary(self, cmd, callback, blocksize=8192, rest=None):
|
||||||
|
"""Retrieve a file in binary mode."""
|
||||||
|
try:
|
||||||
|
if isinstance(cmd, bytes):
|
||||||
|
cmd = cmd.decode(self.encoding)
|
||||||
|
return super().retrbinary(cmd, callback, blocksize, rest)
|
||||||
|
except UnicodeEncodeError:
|
||||||
|
return super().retrbinary(cmd.encode(self.encoding).decode('utf-8'), callback, blocksize, rest)
|
||||||
|
|
||||||
|
|
||||||
# FTP接口类
|
# FTP接口类
|
||||||
@@ -133,7 +180,7 @@ class FtpController:
|
|||||||
|
|
||||||
|
|
||||||
|
|
||||||
def transfer_nc_files(
|
def transfer_files(
|
||||||
source_ftp_info,
|
source_ftp_info,
|
||||||
target_ftp_info,
|
target_ftp_info,
|
||||||
source_dir,
|
source_dir,
|
||||||
@@ -151,7 +198,7 @@ def transfer_nc_files(
|
|||||||
target_dir: str, 目标FTP上的目标目录
|
target_dir: str, 目标FTP上的目标目录
|
||||||
keep_dir: bool, 是否保持目录结构,默认False
|
keep_dir: bool, 是否保持目录结构,默认False
|
||||||
"""
|
"""
|
||||||
trans_status = [False]
|
transfered_file_list = []
|
||||||
try:
|
try:
|
||||||
# 连接源FTP
|
# 连接源FTP
|
||||||
source_ftp = FtpController(
|
source_ftp = FtpController(
|
||||||
@@ -214,52 +261,94 @@ def transfer_nc_files(
|
|||||||
target_path = f"{target_dir}/{relative_path}/{item}"
|
target_path = f"{target_dir}/{relative_path}/{item}"
|
||||||
else:
|
else:
|
||||||
target_path = f"{target_dir}/{item}"
|
target_path = f"{target_dir}/{item}"
|
||||||
with open(temp_path, 'rb') as f:
|
|
||||||
target_ftp.ftp.storbinary(f'STOR {target_path}', f)
|
# 规范化路径
|
||||||
|
target_path = target_path.replace('\\', '/').strip('/')
|
||||||
|
|
||||||
trans_status[0] = True
|
# 确保目标目录存在
|
||||||
|
target_dir_path = '/'.join(target_path.split('/')[:-1])
|
||||||
|
try:
|
||||||
|
target_ftp.ftp.cwd('/') # 回到根目录
|
||||||
|
for dir_part in target_dir_path.split('/'):
|
||||||
|
if dir_part:
|
||||||
|
try:
|
||||||
|
target_ftp.ftp.cwd(dir_part)
|
||||||
|
except:
|
||||||
|
try:
|
||||||
|
target_ftp.ftp.mkd(dir_part)
|
||||||
|
target_ftp.ftp.cwd(dir_part)
|
||||||
|
except Exception as e:
|
||||||
|
logging.error(f"创建目录失败 {dir_part}: {str(e)}")
|
||||||
|
raise
|
||||||
|
except Exception as e:
|
||||||
|
logging.error(f"处理目标目录失败: {str(e)}")
|
||||||
|
raise
|
||||||
|
|
||||||
|
# 检查FTP连接状态
|
||||||
|
try:
|
||||||
|
target_ftp.ftp.voidcmd('NOOP')
|
||||||
|
except:
|
||||||
|
logging.error("FTP连接已断开,尝试重新连接")
|
||||||
|
target_ftp.ftp.connect(target_ftp_info['host'], target_ftp_info['port'])
|
||||||
|
target_ftp.ftp.login(target_ftp_info['username'], target_ftp_info['password'])
|
||||||
|
|
||||||
|
# 上传文件
|
||||||
|
try:
|
||||||
|
with open(temp_path, 'rb') as f:
|
||||||
|
# 检查文件是否可读
|
||||||
|
content = f.read()
|
||||||
|
if not content:
|
||||||
|
raise Exception("临时文件为空")
|
||||||
|
f.seek(0) # 重置文件指针
|
||||||
|
target_ftp.ftp.storbinary(f'STOR {target_path}', f)
|
||||||
|
except Exception as e:
|
||||||
|
logging.error(f"上传文件失败: {str(e)}")
|
||||||
|
logging.error(f"目标路径: {target_path}")
|
||||||
|
raise
|
||||||
|
|
||||||
|
transfered_file_list.append(item)
|
||||||
# 删除临时文件
|
# 删除临时文件
|
||||||
os.remove(temp_path)
|
os.remove(temp_path)
|
||||||
logging.info(f"已传输文件: {item}")
|
logging.info(f"已传输文件: {item}")
|
||||||
|
|
||||||
# 清空目标目录下的所有内容
|
# 清空目标目录下的所有内容
|
||||||
try:
|
# try:
|
||||||
target_ftp.ftp.cwd(target_dir)
|
# target_ftp.ftp.cwd(target_dir)
|
||||||
files = target_ftp.ftp.nlst()
|
# files = target_ftp.ftp.nlst()
|
||||||
|
|
||||||
for f in files:
|
# for f in files:
|
||||||
try:
|
# try:
|
||||||
# 尝试删除文件
|
# # 尝试删除文件
|
||||||
target_ftp.ftp.delete(f)
|
# target_ftp.ftp.delete(f)
|
||||||
except:
|
# except:
|
||||||
try:
|
# try:
|
||||||
# 如果删除失败,可能是目录,递归删除目录
|
# # 如果删除失败,可能是目录,递归删除目录
|
||||||
def remove_dir(path):
|
# def remove_dir(path):
|
||||||
target_ftp.ftp.cwd(path)
|
# target_ftp.ftp.cwd(path)
|
||||||
sub_files = target_ftp.ftp.nlst()
|
# sub_files = target_ftp.ftp.nlst()
|
||||||
for sf in sub_files:
|
# for sf in sub_files:
|
||||||
try:
|
# try:
|
||||||
target_ftp.ftp.delete(sf)
|
# target_ftp.ftp.delete(sf)
|
||||||
except:
|
# except:
|
||||||
remove_dir(f"{path}/{sf}")
|
# remove_dir(f"{path}/{sf}")
|
||||||
target_ftp.ftp.cwd('..')
|
# target_ftp.ftp.cwd('..')
|
||||||
target_ftp.ftp.rmd(path)
|
# target_ftp.ftp.rmd(path)
|
||||||
|
|
||||||
remove_dir(f"{target_dir}/{f}")
|
# remove_dir(f"{target_dir}/{f}")
|
||||||
except:
|
# except:
|
||||||
logging.error(f"无法删除 {f}")
|
# logging.error(f"无法删除 {f}")
|
||||||
pass
|
# pass
|
||||||
|
|
||||||
logging.info(f"已清空目标目录 {target_dir}")
|
# logging.info(f"已清空目标目录 {target_dir}")
|
||||||
except Exception as e:
|
# except Exception as e:
|
||||||
logging.error(f"清空目标目录失败: {str(e)}")
|
# logging.error(f"清空目标目录失败: {str(e)}")
|
||||||
raise Exception(f"清空目标目录失败: {str(e)}")
|
# raise Exception(f"清空目标目录失败: {str(e)}")
|
||||||
|
|
||||||
# 开始遍历
|
# 开始遍历
|
||||||
traverse_dir(source_dir)
|
traverse_dir(source_dir)
|
||||||
|
|
||||||
logging.info("所有文件传输完成")
|
logging.info("所有文件传输完成")
|
||||||
return trans_status[0]
|
return transfered_file_list
|
||||||
|
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
logging.error(f"传输过程出错: {str(e)}")
|
logging.error(f"传输过程出错: {str(e)}")
|
||||||
|
|||||||
@@ -2,6 +2,8 @@
|
|||||||
import json
|
import json
|
||||||
import base64
|
import base64
|
||||||
import logging
|
import logging
|
||||||
|
import qrcode
|
||||||
|
from io import BytesIO
|
||||||
from datetime import timedelta
|
from datetime import timedelta
|
||||||
import requests
|
import requests
|
||||||
from odoo.addons.sf_base.commons.common import Common
|
from odoo.addons.sf_base.commons.common import Common
|
||||||
@@ -34,8 +36,7 @@ class SfMaintenanceEquipmentAGVLog(models.Model):
|
|||||||
|
|
||||||
|
|
||||||
class SfMaintenanceEquipment(models.Model):
|
class SfMaintenanceEquipment(models.Model):
|
||||||
_name = 'maintenance.equipment'
|
_inherit = 'maintenance.equipment'
|
||||||
_inherit = ['maintenance.equipment', 'printing.utils']
|
|
||||||
_description = '设备'
|
_description = '设备'
|
||||||
|
|
||||||
crea_url = "/api/machine_tool/create"
|
crea_url = "/api/machine_tool/create"
|
||||||
@@ -844,7 +845,7 @@ class SfMaintenanceEquipment(models.Model):
|
|||||||
box_size=10,
|
box_size=10,
|
||||||
border=4,
|
border=4,
|
||||||
)
|
)
|
||||||
qr.add_data(record.id)
|
qr.add_data(record.MTcode)
|
||||||
qr.make(fit=True)
|
qr.make(fit=True)
|
||||||
qr_image = qr.make_image(fill_color="black", back_color="white")
|
qr_image = qr.make_image(fill_color="black", back_color="white")
|
||||||
|
|
||||||
@@ -855,66 +856,6 @@ class SfMaintenanceEquipment(models.Model):
|
|||||||
|
|
||||||
record.qr_code_image = encoded_image
|
record.qr_code_image = encoded_image
|
||||||
|
|
||||||
def print_single_method(self):
|
|
||||||
|
|
||||||
print('self.name========== %s' % self.name)
|
|
||||||
self.ensure_one()
|
|
||||||
qr_code_data = self.qr_code_image
|
|
||||||
if not qr_code_data:
|
|
||||||
raise UserError("没有找到二维码数据。")
|
|
||||||
maintenance_equipment_name = self.name
|
|
||||||
# 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_name, host, port)
|
|
||||||
|
|
||||||
|
|
||||||
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
|
|
||||||
|
|
||||||
|
|
||||||
class SfRobotAxisNum(models.Model):
|
class SfRobotAxisNum(models.Model):
|
||||||
_name = 'sf.robot.axis.num'
|
_name = 'sf.robot.axis.num'
|
||||||
|
|||||||
@@ -1055,11 +1055,6 @@
|
|||||||
|
|
||||||
<xpath expr="//group/field[@name='location']" position="after">
|
<xpath expr="//group/field[@name='location']" position="after">
|
||||||
<field name="qr_code_image" widget="image" readonly="1" attrs="{'invisible': [('equipment_type', '!=', '机床')]}" />
|
<field name="qr_code_image" widget="image" readonly="1" attrs="{'invisible': [('equipment_type', '!=', '机床')]}" />
|
||||||
<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>
|
</xpath>
|
||||||
<xpath expr="//page[@name='maintenance']" position="after">
|
<xpath expr="//page[@name='maintenance']" position="after">
|
||||||
<page name="network_config" string="网络配置" attrs="{'invisible': [('equipment_type', '!=', '机床')]}" >
|
<page name="network_config" string="网络配置" attrs="{'invisible': [('equipment_type', '!=', '机床')]}" >
|
||||||
|
|||||||
@@ -48,6 +48,7 @@
|
|||||||
'views/mrp_workorder_batch_replan.xml',
|
'views/mrp_workorder_batch_replan.xml',
|
||||||
'views/purchase_order_view.xml',
|
'views/purchase_order_view.xml',
|
||||||
'views/product_template_views.xml',
|
'views/product_template_views.xml',
|
||||||
|
# 'views/stock_warehouse_orderpoint.xml',
|
||||||
],
|
],
|
||||||
'assets': {
|
'assets': {
|
||||||
|
|
||||||
|
|||||||
@@ -6,12 +6,14 @@ from datetime import datetime
|
|||||||
from odoo.addons.sf_manufacturing.models.agv_scheduling import RepeatTaskException
|
from odoo.addons.sf_manufacturing.models.agv_scheduling import RepeatTaskException
|
||||||
from odoo import http
|
from odoo import http
|
||||||
from odoo.http import request
|
from odoo.http import request
|
||||||
|
from odoo.addons.sf_base.decorators.api_log import api_log
|
||||||
|
|
||||||
|
|
||||||
class Manufacturing_Connect(http.Controller):
|
class Manufacturing_Connect(http.Controller):
|
||||||
|
|
||||||
@http.route('/AutoDeviceApi/GetWoInfo', type='json', auth='sf_token', methods=['GET', 'POST'], csrf=False,
|
@http.route('/AutoDeviceApi/GetWoInfo', type='json', auth='sf_token', methods=['GET', 'POST'], csrf=False,
|
||||||
cors="*")
|
cors="*")
|
||||||
|
@api_log('获取工单', requester='中控系统')
|
||||||
def get_Work_Info(self, **kw):
|
def get_Work_Info(self, **kw):
|
||||||
"""
|
"""
|
||||||
自动化传递工单号获取工单信息
|
自动化传递工单号获取工单信息
|
||||||
@@ -54,6 +56,7 @@ class Manufacturing_Connect(http.Controller):
|
|||||||
|
|
||||||
@http.route('/AutoDeviceApi/GetShiftPlan', type='json', auth='sf_token', methods=['GET', 'POST'], csrf=False,
|
@http.route('/AutoDeviceApi/GetShiftPlan', type='json', auth='sf_token', methods=['GET', 'POST'], csrf=False,
|
||||||
cors="*")
|
cors="*")
|
||||||
|
@api_log('获取日计划', requester='中控系统')
|
||||||
def get_ShiftPlan(self, **kw):
|
def get_ShiftPlan(self, **kw):
|
||||||
"""
|
"""
|
||||||
自动化每天获取机台日计划
|
自动化每天获取机台日计划
|
||||||
@@ -107,6 +110,7 @@ class Manufacturing_Connect(http.Controller):
|
|||||||
|
|
||||||
@http.route('/AutoDeviceApi/QcCheck', type='json', auth='sf_token', methods=['GET', 'POST'], csrf=False,
|
@http.route('/AutoDeviceApi/QcCheck', type='json', auth='sf_token', methods=['GET', 'POST'], csrf=False,
|
||||||
cors="*")
|
cors="*")
|
||||||
|
@api_log('工件预调(前置三元检测)', requester='中控系统')
|
||||||
def get_qcCheck(self, **kw):
|
def get_qcCheck(self, **kw):
|
||||||
"""
|
"""
|
||||||
工件预调(前置三元检测)
|
工件预调(前置三元检测)
|
||||||
@@ -149,6 +153,7 @@ class Manufacturing_Connect(http.Controller):
|
|||||||
|
|
||||||
@http.route('/AutoDeviceApi/FeedBackStart', type='json', auth='none', methods=['GET', 'POST'], csrf=False,
|
@http.route('/AutoDeviceApi/FeedBackStart', type='json', auth='none', methods=['GET', 'POST'], csrf=False,
|
||||||
cors="*")
|
cors="*")
|
||||||
|
@api_log('工单开始', requester='中控系统')
|
||||||
def button_Work_START(self, **kw):
|
def button_Work_START(self, **kw):
|
||||||
"""
|
"""
|
||||||
工单任务开始
|
工单任务开始
|
||||||
@@ -198,6 +203,7 @@ class Manufacturing_Connect(http.Controller):
|
|||||||
|
|
||||||
@http.route('/AutoDeviceApi/FeedBackEnd', type='json', auth='none', methods=['GET', 'POST'], csrf=False,
|
@http.route('/AutoDeviceApi/FeedBackEnd', type='json', auth='none', methods=['GET', 'POST'], csrf=False,
|
||||||
cors="*")
|
cors="*")
|
||||||
|
@api_log('工单结束', requester='中控系统')
|
||||||
def button_Work_End(self, **kw):
|
def button_Work_End(self, **kw):
|
||||||
"""
|
"""
|
||||||
工单任务结束
|
工单任务结束
|
||||||
@@ -249,6 +255,7 @@ class Manufacturing_Connect(http.Controller):
|
|||||||
|
|
||||||
@http.route('/AutoDeviceApi/PartQualityInspect', type='json', auth='none', methods=['GET', 'POST'], csrf=False,
|
@http.route('/AutoDeviceApi/PartQualityInspect', type='json', auth='none', methods=['GET', 'POST'], csrf=False,
|
||||||
cors="*")
|
cors="*")
|
||||||
|
@api_log('零件检测(后置三元检测)', requester='中控系统')
|
||||||
def PartQualityInspect(self, **kw):
|
def PartQualityInspect(self, **kw):
|
||||||
"""
|
"""
|
||||||
零件质检
|
零件质检
|
||||||
@@ -295,6 +302,7 @@ class Manufacturing_Connect(http.Controller):
|
|||||||
|
|
||||||
@http.route('/AutoDeviceApi/CMMProgDolod', type='json', auth='none', methods=['GET', 'POST'], csrf=False,
|
@http.route('/AutoDeviceApi/CMMProgDolod', type='json', auth='none', methods=['GET', 'POST'], csrf=False,
|
||||||
cors="*")
|
cors="*")
|
||||||
|
@api_log('CMM测量程序下载', requester='中控系统')
|
||||||
def CMMProgDolod(self, **kw):
|
def CMMProgDolod(self, **kw):
|
||||||
"""
|
"""
|
||||||
中控系统传递RFID编号给MES,获取测量程序文件。Ftp下载文件
|
中控系统传递RFID编号给MES,获取测量程序文件。Ftp下载文件
|
||||||
@@ -335,6 +343,7 @@ class Manufacturing_Connect(http.Controller):
|
|||||||
|
|
||||||
@http.route('/AutoDeviceApi/NCProgDolod', type='json', auth='none', methods=['GET', 'POST'], csrf=False,
|
@http.route('/AutoDeviceApi/NCProgDolod', type='json', auth='none', methods=['GET', 'POST'], csrf=False,
|
||||||
cors="*")
|
cors="*")
|
||||||
|
@api_log('CAM加工程序下载', requester='中控系统')
|
||||||
def NCProgDolod(self, **kw):
|
def NCProgDolod(self, **kw):
|
||||||
"""
|
"""
|
||||||
中控系统传递RFID编号给MES,获取程序单及程序文件。Ftp下载文件
|
中控系统传递RFID编号给MES,获取程序单及程序文件。Ftp下载文件
|
||||||
@@ -376,6 +385,7 @@ class Manufacturing_Connect(http.Controller):
|
|||||||
|
|
||||||
@http.route('/AutoDeviceApi/LocationChange', type='json', auth='sf_token', methods=['GET', 'POST'], csrf=False,
|
@http.route('/AutoDeviceApi/LocationChange', type='json', auth='sf_token', methods=['GET', 'POST'], csrf=False,
|
||||||
cors="*")
|
cors="*")
|
||||||
|
@api_log('库位变更', requester='中控系统')
|
||||||
def LocationChange(self, **kw):
|
def LocationChange(self, **kw):
|
||||||
"""
|
"""
|
||||||
库位变更
|
库位变更
|
||||||
@@ -435,32 +445,7 @@ class Manufacturing_Connect(http.Controller):
|
|||||||
shelfinfo = list(filter(lambda x: x.get('DeviceId') == DeciveId,
|
shelfinfo = list(filter(lambda x: x.get('DeviceId') == DeciveId,
|
||||||
request.env['sf.shelf.location'].sudo().get_sf_shelf_location_info(
|
request.env['sf.shelf.location'].sudo().get_sf_shelf_location_info(
|
||||||
DeciveId)))
|
DeciveId)))
|
||||||
total_data = request.env['sf.shelf.location.datasync'].sudo().get_total_data()
|
request.env['sf.shelf.location.datasync'].sudo().set_shelf_location(shelfinfo)
|
||||||
for item in shelfinfo:
|
|
||||||
logging.info('货架已获取信息:%s' % item)
|
|
||||||
shelf_barcode = request.env['sf.shelf.location.datasync'].sudo().find_our_code(
|
|
||||||
total_data, item['Postion'])
|
|
||||||
location_id = request.env['sf.shelf.location'].sudo().search(
|
|
||||||
[('barcode', '=', shelf_barcode)],
|
|
||||||
limit=1)
|
|
||||||
if location_id:
|
|
||||||
# 如果是线边刀库信息,则对功能刀具移动生成记录
|
|
||||||
if 'Tool' in item['Postion']:
|
|
||||||
tool = request.env['sf.functional.cutting.tool.entity'].sudo().search(
|
|
||||||
[('rfid', '=', item['RfidCode']), ('functional_tool_status', '!=', '已拆除')])
|
|
||||||
tool.sudo().tool_in_out_stock_location(location_id)
|
|
||||||
if tool:
|
|
||||||
location_id.product_sn_id = tool.barcode_id.id
|
|
||||||
# 修改功能刀具状态
|
|
||||||
if item.get('State') == '报警':
|
|
||||||
if tool.functional_tool_status != item.get('State'):
|
|
||||||
tool.write({
|
|
||||||
'functional_tool_status': item['State']
|
|
||||||
})
|
|
||||||
else:
|
|
||||||
location_id.product_sn_id = False
|
|
||||||
if item['RfidCode']:
|
|
||||||
logging.info('Rfid为【%s】的功能刀具在系统中不存在!' % item['RfidCode'])
|
|
||||||
else:
|
else:
|
||||||
equipment_id = request.env['maintenance.equipment'].sudo().search([('name', '=', DeciveId)])
|
equipment_id = request.env['maintenance.equipment'].sudo().search([('name', '=', DeciveId)])
|
||||||
if equipment_id:
|
if equipment_id:
|
||||||
@@ -480,6 +465,7 @@ class Manufacturing_Connect(http.Controller):
|
|||||||
|
|
||||||
@http.route('/AutoDeviceApi/AGVToProduct', type='json', auth='none', methods=['GET', 'POST'], csrf=False,
|
@http.route('/AutoDeviceApi/AGVToProduct', type='json', auth='none', methods=['GET', 'POST'], csrf=False,
|
||||||
cors="*")
|
cors="*")
|
||||||
|
@api_log('AGV运送上产线', requester='中控系统')
|
||||||
def AGVToProduct(self, **kw):
|
def AGVToProduct(self, **kw):
|
||||||
"""
|
"""
|
||||||
AGV运送上产线(完成)
|
AGV运送上产线(完成)
|
||||||
@@ -552,6 +538,7 @@ class Manufacturing_Connect(http.Controller):
|
|||||||
|
|
||||||
@http.route('/AutoDeviceApi/AGVDownProduct', type='json', auth='none', methods=['GET', 'POST'], csrf=False,
|
@http.route('/AutoDeviceApi/AGVDownProduct', type='json', auth='none', methods=['GET', 'POST'], csrf=False,
|
||||||
cors="*")
|
cors="*")
|
||||||
|
@api_log('AGV运送下产线', requester='中控系统')
|
||||||
def AGVDownProduct(self, **kw):
|
def AGVDownProduct(self, **kw):
|
||||||
"""
|
"""
|
||||||
MES调度AGV,搬运零件AGV托盘到产线接驳站。
|
MES调度AGV,搬运零件AGV托盘到产线接驳站。
|
||||||
|
|||||||
@@ -27,7 +27,7 @@ class JikimoSaleRoutePicking(Sf_Bf_Connect):
|
|||||||
bfm_process_order_list = json.loads(kw['bfm_process_order_list'])
|
bfm_process_order_list = json.loads(kw['bfm_process_order_list'])
|
||||||
order_id = request.env['sale.order'].with_user(request.env.ref("base.user_admin")).sale_order_create(
|
order_id = request.env['sale.order'].with_user(request.env.ref("base.user_admin")).sale_order_create(
|
||||||
company_id, kw['delivery_name'], kw['delivery_telephone'], kw['delivery_address'],
|
company_id, kw['delivery_name'], kw['delivery_telephone'], kw['delivery_address'],
|
||||||
kw['delivery_end_date'], kw['payments_way'], kw['pay_way'], kw['order_number'], state='draft',
|
kw['delivery_end_date'], kw['payments_way'], kw['pay_way'], kw['order_number'], kw['remark'], state='draft',
|
||||||
model_display_version=kw.get('model_display_version'))
|
model_display_version=kw.get('model_display_version'))
|
||||||
i = 1
|
i = 1
|
||||||
# 给sale_order的default_code字段赋值
|
# 给sale_order的default_code字段赋值
|
||||||
@@ -45,6 +45,8 @@ class JikimoSaleRoutePicking(Sf_Bf_Connect):
|
|||||||
product.product_tmpl_id.is_customer_provided = True if item['embryo_redundancy_id'] else False
|
product.product_tmpl_id.is_customer_provided = True if item['embryo_redundancy_id'] else False
|
||||||
order_id.with_user(request.env.ref("base.user_admin")).sale_order_create_line(product, item)
|
order_id.with_user(request.env.ref("base.user_admin")).sale_order_create_line(product, item)
|
||||||
i += 1
|
i += 1
|
||||||
|
# BFM 内部下单 新增合同等内容补充
|
||||||
|
order_id.write_sale_documents(kw)
|
||||||
res['factory_order_no'] = order_id.name
|
res['factory_order_no'] = order_id.name
|
||||||
order_id.confirm_to_supply_method()
|
order_id.confirm_to_supply_method()
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
|
|||||||
@@ -18,4 +18,4 @@ from . import quick_easy_order
|
|||||||
from . import purchase_order
|
from . import purchase_order
|
||||||
from . import quality_check
|
from . import quality_check
|
||||||
from . import purchase_request_line
|
from . import purchase_request_line
|
||||||
from . import workorder_printer
|
# from . import stock_warehouse_orderpoint
|
||||||
@@ -279,10 +279,12 @@ class MrpProduction(models.Model):
|
|||||||
production_id.part_name = production_id.product_id.part_name
|
production_id.part_name = production_id.product_id.part_name
|
||||||
elif production_id.product_id.categ_id.type == '坯料':
|
elif production_id.product_id.categ_id.type == '坯料':
|
||||||
product_name = ''
|
product_name = ''
|
||||||
match = re.search(r'(S\d{5}-\d)', production_id.product_id.name)
|
match = re.search(r'(S\d{5}-\d+)', production_id.product_id.name)
|
||||||
# 如果匹配成功,提取结果
|
# 如果匹配成功,提取结果
|
||||||
if match:
|
if match:
|
||||||
product_name = match.group(0)
|
product_name = match.group(0)
|
||||||
|
else:
|
||||||
|
product_name = production_id.product_id.name
|
||||||
if production_id.sale_order_id:
|
if production_id.sale_order_id:
|
||||||
sale_order = production_id.sale_order_id
|
sale_order = production_id.sale_order_id
|
||||||
else:
|
else:
|
||||||
@@ -636,13 +638,17 @@ class MrpProduction(models.Model):
|
|||||||
# 增加触发时间参数
|
# 增加触发时间参数
|
||||||
def update_programming_state(self, trigger_time=None, reprogramming_reason=None):
|
def update_programming_state(self, trigger_time=None, reprogramming_reason=None):
|
||||||
try:
|
try:
|
||||||
|
reason = ""
|
||||||
manufacturing_type = None
|
manufacturing_type = None
|
||||||
if self.is_scrap:
|
if self.is_scrap:
|
||||||
manufacturing_type = 'scrap'
|
manufacturing_type = 'scrap'
|
||||||
|
reason = "报废"
|
||||||
elif self.tool_state == '2':
|
elif self.tool_state == '2':
|
||||||
manufacturing_type = 'invalid_tool_rework'
|
manufacturing_type = 'invalid_tool_rework'
|
||||||
|
reason = "无效功能刀具"
|
||||||
elif self.is_rework:
|
elif self.is_rework:
|
||||||
manufacturing_type = 'rework'
|
manufacturing_type = 'rework'
|
||||||
|
reason = "返工"
|
||||||
res = {'programming_no': self.programming_no,
|
res = {'programming_no': self.programming_no,
|
||||||
'manufacturing_type': manufacturing_type,
|
'manufacturing_type': manufacturing_type,
|
||||||
'trigger_time': trigger_time,
|
'trigger_time': trigger_time,
|
||||||
@@ -657,6 +663,16 @@ class MrpProduction(models.Model):
|
|||||||
result = json.loads(ret['result'])
|
result = json.loads(ret['result'])
|
||||||
logging.info('update_programming_state-ret:%s' % result)
|
logging.info('update_programming_state-ret:%s' % result)
|
||||||
if result['status'] == 1:
|
if result['status'] == 1:
|
||||||
|
self.programming_record_ids.create({
|
||||||
|
'number': len(self.programming_record_ids) + 1,
|
||||||
|
'production_id': self.id,
|
||||||
|
'reason': reason,
|
||||||
|
'programming_method': False,
|
||||||
|
'current_programming_count': False,
|
||||||
|
'target_production_id': False,
|
||||||
|
'apply_time': fields.Datetime.now(),
|
||||||
|
'send_time': False,
|
||||||
|
})
|
||||||
self.write({'is_rework': True})
|
self.write({'is_rework': True})
|
||||||
else:
|
else:
|
||||||
raise UserError(ret['message'])
|
raise UserError(ret['message'])
|
||||||
@@ -787,6 +803,17 @@ class MrpProduction(models.Model):
|
|||||||
if ret['status'] == 1:
|
if ret['status'] == 1:
|
||||||
self.write(
|
self.write(
|
||||||
{'programming_no': ret['programming_no'], 'programming_state': '编程中', 'work_state': '编程中'})
|
{'programming_no': ret['programming_no'], 'programming_state': '编程中', 'work_state': '编程中'})
|
||||||
|
# 生成编程记录
|
||||||
|
self.programming_record_ids.create({
|
||||||
|
'number': len(self.programming_record_ids) + 1,
|
||||||
|
'production_id': self.id,
|
||||||
|
'reason': '首次下发',
|
||||||
|
'programming_method': False,
|
||||||
|
'current_programming_count': False,
|
||||||
|
'target_production_id': False,
|
||||||
|
'apply_time': fields.Datetime.now(),
|
||||||
|
'send_time': False,
|
||||||
|
})
|
||||||
else:
|
else:
|
||||||
raise UserError(ret['message'])
|
raise UserError(ret['message'])
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
@@ -900,40 +927,41 @@ class MrpProduction(models.Model):
|
|||||||
for workorder in production.workorder_ids:
|
for workorder in production.workorder_ids:
|
||||||
workorder.duration_expected = workorder._get_duration_expected()
|
workorder.duration_expected = workorder._get_duration_expected()
|
||||||
|
|
||||||
def _create_subcontract_purchase_request(self, purchase_request_line):
|
# def _create_subcontract_purchase_request(self, purchase_request_line):
|
||||||
sorted_list = sorted(purchase_request_line, key=itemgetter('name'))
|
# sorted_list = sorted(purchase_request_line, key=itemgetter('name'))
|
||||||
grouped_purchase_request_line = {
|
# grouped_purchase_request_line = {
|
||||||
k: list(g)
|
# k: list(g)
|
||||||
for k, g in groupby(sorted_list, key=itemgetter('name'))
|
# for k, g in groupby(sorted_list, key=itemgetter('name'))
|
||||||
}
|
# }
|
||||||
for name, request_line in grouped_purchase_request_line.items():
|
# for name, request_line in grouped_purchase_request_line.items():
|
||||||
request_line_sorted_list = sorted(request_line, key=itemgetter('product_id'))
|
# request_line_sorted_list = sorted(request_line, key=itemgetter('product_id'))
|
||||||
grouped_purchase_request_line_sorted_list = {
|
# grouped_purchase_request_line_sorted_list = {
|
||||||
k: list(g)
|
# k: list(g)
|
||||||
for k, g in groupby(request_line_sorted_list, key=itemgetter('product_id'))
|
# for k, g in groupby(request_line_sorted_list, key=itemgetter('product_id'))
|
||||||
}
|
# }
|
||||||
purchase_request_model = self.env["purchase.request"]
|
# purchase_request_model = self.env["purchase.request"]
|
||||||
origin = ", ".join({item['production_name'] for item in request_line_sorted_list if item.get('production_name')})
|
# origin = ", ".join({item['production_name'] for item in request_line_sorted_list if item.get('production_name')})
|
||||||
pr = purchase_request_model.create({
|
# pr = purchase_request_model.create({
|
||||||
"origin": origin,
|
# "origin": origin,
|
||||||
"company_id": self.company_id.id,
|
# "company_id": self.company_id.id,
|
||||||
"picking_type_id": self.env.ref('stock.picking_type_in').id,
|
# "picking_type_id": self.env.ref('stock.picking_type_in').id,
|
||||||
"group_id": request_line[0].get('group_id'),
|
# "group_id": request_line[0].get('group_id'),
|
||||||
"requested_by": self.env.context.get("uid", self.env.uid),
|
# "requested_by": self.env.context.get("uid", self.env.uid),
|
||||||
"assigned_to": False,
|
# "assigned_to": False,
|
||||||
"bom_id": self[0].bom_id.id,
|
# "bom_id": self[0].bom_id.id,
|
||||||
"is_subcontract":True,
|
# "is_subcontract":True,
|
||||||
})
|
# })
|
||||||
self[0].bom_id.bom_line_ids.product_id.route_ids = [(4,self.env.ref(
|
# self[0].bom_id.bom_line_ids.product_id.route_ids = [(4,self.env.ref(
|
||||||
'sf_stock.stock_route_process_outsourcing').id)]
|
# 'sf_stock.stock_route_process_outsourcing').id)]
|
||||||
for product_id, request_line_list in grouped_purchase_request_line_sorted_list.items():
|
# for product_id, request_line_list in grouped_purchase_request_line_sorted_list.items():
|
||||||
cur_request_line = request_line_list[0]
|
# cur_request_line = request_line_list[0]
|
||||||
cur_request_line['product_qty'] = len(request_line_list)
|
# # cur_request_line['product_qty'] = cur_request_line['product_qty']
|
||||||
cur_request_line['request_id'] = pr.id
|
# cur_request_line['request_id'] = pr.id
|
||||||
cur_request_line['origin'] = ", ".join({item['production_name'] for item in request_line_list if item.get('production_name')})
|
# cur_request_line['origin'] = ", ".join({item['production_name'] for item in request_line_list if item.get('production_name')})
|
||||||
cur_request_line.pop('group_id', None)
|
# cur_request_line.pop('group_id', None)
|
||||||
cur_request_line.pop('production_name', None)
|
# cur_request_line.pop('production_name', None)
|
||||||
self.env["purchase.request.line"].create(cur_request_line)
|
# self.env["purchase.request.line"].create(cur_request_line)
|
||||||
|
# pr.button_approved()
|
||||||
|
|
||||||
# 外协出入库单处理
|
# 外协出入库单处理
|
||||||
def get_subcontract_pick_purchase(self):
|
def get_subcontract_pick_purchase(self):
|
||||||
@@ -962,13 +990,13 @@ class MrpProduction(models.Model):
|
|||||||
return
|
return
|
||||||
for workorders in reversed(sorted_workorders):
|
for workorders in reversed(sorted_workorders):
|
||||||
self.env['stock.picking'].create_outcontract_picking(workorders, production, sorted_workorders)
|
self.env['stock.picking'].create_outcontract_picking(workorders, production, sorted_workorders)
|
||||||
# self.env['purchase.order'].get_purchase_order(workorders, production, product_id_to_production_names)
|
self.env['purchase.order'].get_purchase_order(workorders, production, product_id_to_production_names)
|
||||||
purchase_request_line = purchase_request_line + self.env['purchase.order'].get_purchase_request(
|
# purchase_request_line = purchase_request_line + self.env['purchase.order'].get_purchase_request(
|
||||||
workorders, production)
|
# workorders, production)
|
||||||
all_workorders += workorders
|
# all_workorders += workorders
|
||||||
self._create_subcontract_purchase_request(purchase_request_line)
|
# self._create_subcontract_purchase_request(purchase_request_line)
|
||||||
for workorder in all_workorders:
|
# for workorder in all_workorders:
|
||||||
workorder._compute_pr_mp_count()
|
# workorder._compute_pr_mp_count()
|
||||||
# 工单排序
|
# 工单排序
|
||||||
def _reset_work_order_sequence1(self, k):
|
def _reset_work_order_sequence1(self, k):
|
||||||
for rec in self:
|
for rec in self:
|
||||||
@@ -1771,7 +1799,6 @@ class MrpProduction(models.Model):
|
|||||||
"""
|
"""
|
||||||
检查前置条件:制造订单【状态】=“待排程、待加工”,制造订单的【编程状态】=“已编程”。
|
检查前置条件:制造订单【状态】=“待排程、待加工”,制造订单的【编程状态】=“已编程”。
|
||||||
"""
|
"""
|
||||||
print('申请编程')
|
|
||||||
if len(self) > 1:
|
if len(self) > 1:
|
||||||
raise UserError('仅支持选择单个制造订单进行编程申请,请重新选择')
|
raise UserError('仅支持选择单个制造订单进行编程申请,请重新选择')
|
||||||
for production in self:
|
for production in self:
|
||||||
@@ -1821,7 +1848,7 @@ class MrpProduction(models.Model):
|
|||||||
logging.info('update_programming_state error:%s' % e)
|
logging.info('update_programming_state error:%s' % e)
|
||||||
raise UserError("更新编程单状态失败,请联系管理员")
|
raise UserError("更新编程单状态失败,请联系管理员")
|
||||||
|
|
||||||
model_id = fields.Char('模型id', related='product_id.model_id')
|
model_id = fields.Char('模型ID', related='product_id.model_id')
|
||||||
|
|
||||||
|
|
||||||
# 编程记录
|
# 编程记录
|
||||||
@@ -1840,6 +1867,7 @@ class sf_programming_record(models.Model):
|
|||||||
target_production_id = fields.Char('目标制造单号')
|
target_production_id = fields.Char('目标制造单号')
|
||||||
apply_time = fields.Datetime('申请时间')
|
apply_time = fields.Datetime('申请时间')
|
||||||
send_time = fields.Datetime('下发时间')
|
send_time = fields.Datetime('下发时间')
|
||||||
|
apply_uid = fields.Many2one('res.users', '申请人', default=lambda self: self.env.user)
|
||||||
|
|
||||||
|
|
||||||
class sf_detection_result(models.Model):
|
class sf_detection_result(models.Model):
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
import logging
|
import logging
|
||||||
from odoo import fields, models, api
|
from odoo import fields, models, api
|
||||||
from odoo.exceptions import UserError
|
from odoo.exceptions import UserError
|
||||||
from odoo.tools import str2bool
|
# from odoo.tools import str2bool
|
||||||
|
|
||||||
|
|
||||||
class ResMrpRoutingWorkcenter(models.Model):
|
class ResMrpRoutingWorkcenter(models.Model):
|
||||||
@@ -25,20 +25,20 @@ class ResMrpRoutingWorkcenter(models.Model):
|
|||||||
workcenter_ids = fields.Many2many('mrp.workcenter', 'rel_workcenter_route', required=True)
|
workcenter_ids = fields.Many2many('mrp.workcenter', 'rel_workcenter_route', required=True)
|
||||||
bom_id = fields.Many2one('mrp.bom', required=False)
|
bom_id = fields.Many2one('mrp.bom', required=False)
|
||||||
surface_technics_id = fields.Many2one('sf.production.process', string="表面工艺")
|
surface_technics_id = fields.Many2one('sf.production.process', string="表面工艺")
|
||||||
optional_process_parameters = fields.One2many('sf.production.process.parameter','routing_id',string='可选工艺参数')
|
# optional_process_parameters = fields.One2many('sf.production.process.parameter','routing_id',string='可选工艺参数')
|
||||||
reserved_duration = fields.Float('预留时长', default=30, tracking=True)
|
reserved_duration = fields.Float('预留时长', default=30, tracking=True)
|
||||||
is_outsource = fields.Boolean('外协', default=False)
|
is_outsource = fields.Boolean('外协', default=False)
|
||||||
individuation_page_ids = fields.Many2many('sf.work.individuation.page', string='个性化记录')
|
individuation_page_ids = fields.Many2many('sf.work.individuation.page', string='个性化记录')
|
||||||
|
|
||||||
@api.onchange('surface_technics_id')
|
# @api.onchange('surface_technics_id')
|
||||||
def optional_process_parameters_date(self):
|
# def optional_process_parameters_date(self):
|
||||||
for record in self:
|
# for record in self:
|
||||||
if not record.surface_technics_id:
|
# if not record.surface_technics_id:
|
||||||
continue
|
# continue
|
||||||
parameter_ids = self.env['sf.production.process.parameter'].search([
|
# parameter_ids = self.env['sf.production.process.parameter'].search([
|
||||||
('process_id', '=', record.surface_technics_id.id),
|
# ('process_id', '=', record.surface_technics_id.id),
|
||||||
])
|
# ])
|
||||||
record.optional_process_parameters = parameter_ids.ids
|
# record.optional_process_parameters = parameter_ids.ids
|
||||||
|
|
||||||
# @api.model
|
# @api.model
|
||||||
# def _auto_init(self):
|
# def _auto_init(self):
|
||||||
|
|||||||
@@ -21,16 +21,16 @@ class ResWorkcenter(models.Model):
|
|||||||
related='equipment_id.production_line_id', store=True)
|
related='equipment_id.production_line_id', store=True)
|
||||||
is_process_outsourcing = fields.Boolean('工艺外协')
|
is_process_outsourcing = fields.Boolean('工艺外协')
|
||||||
users_ids = fields.Many2many("res.users", 'users_workcenter', tracking=True)
|
users_ids = fields.Many2many("res.users", 'users_workcenter', tracking=True)
|
||||||
@api.constrains('name')
|
# @api.constrains('name')
|
||||||
def _check_unique_name_code(self):
|
# def _check_unique_name_code(self):
|
||||||
for record in self:
|
# for record in self:
|
||||||
# 检查是否已经存在相同的 name 和 code 组合
|
# # 检查是否已经存在相同的 name 和 code 组合
|
||||||
existing = self.search([
|
# existing = self.search([
|
||||||
('name', '=', record.name),
|
# ('name', '=', record.name),
|
||||||
('id', '!=', record.id) # 排除当前记录
|
# ('id', '!=', record.id) # 排除当前记录
|
||||||
])
|
# ])
|
||||||
if existing:
|
# if existing:
|
||||||
raise ValueError('记录已存在')
|
# raise ValueError('记录已存在')
|
||||||
def write(self, vals):
|
def write(self, vals):
|
||||||
if 'users_ids' in vals:
|
if 'users_ids' in vals:
|
||||||
old_users = self.users_ids
|
old_users = self.users_ids
|
||||||
|
|||||||
@@ -70,21 +70,22 @@ class ResMrpWorkOrder(models.Model):
|
|||||||
delivery_warning = fields.Selection([('normal', '正常'), ('warning', '告警'), ('overdue', '逾期')], string='时效',
|
delivery_warning = fields.Selection([('normal', '正常'), ('warning', '告警'), ('overdue', '逾期')], string='时效',
|
||||||
tracking=True)
|
tracking=True)
|
||||||
back_button_display = fields.Boolean(default=False, compute='_compute_back_button_display', store=True)
|
back_button_display = fields.Boolean(default=False, compute='_compute_back_button_display', store=True)
|
||||||
pr_mp_count = fields.Integer('采购申请单数量', compute='_compute_pr_mp_count', store=True)
|
# 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 not item.is_subcontract:
|
||||||
|
# item.pr_mp_count = 0
|
||||||
|
# continue
|
||||||
|
# pr_ids = self.env['purchase.request'].sudo().search(
|
||||||
|
# [('origin', 'like', item.production_id.name), ('is_subcontract', '=', 'True'),
|
||||||
|
# ('state', '!=', 'rejected')])
|
||||||
|
# if pr_ids:
|
||||||
|
# item.pr_mp_count = len(pr_ids)
|
||||||
|
# else:
|
||||||
|
# item.pr_mp_count = 0
|
||||||
|
|
||||||
@api.depends('state')
|
|
||||||
def _compute_pr_mp_count(self):
|
|
||||||
for item in self:
|
|
||||||
if not item.is_subcontract:
|
|
||||||
item.pr_mp_count = 0
|
|
||||||
continue
|
|
||||||
pr_ids = self.env['purchase.request'].sudo().search(
|
|
||||||
[('origin', 'like', item.production_id.name), ('is_subcontract', '=', 'True'),
|
|
||||||
('state', '!=', 'rejected')])
|
|
||||||
if pr_ids:
|
|
||||||
item.pr_mp_count = len(pr_ids)
|
|
||||||
else:
|
|
||||||
item.pr_mp_count = 0
|
|
||||||
@api.depends('state')
|
@api.depends('state')
|
||||||
def _compute_back_button_display(self):
|
def _compute_back_button_display(self):
|
||||||
for record in self:
|
for record in self:
|
||||||
@@ -129,8 +130,14 @@ class ResMrpWorkOrder(models.Model):
|
|||||||
record.back_button_display = False
|
record.back_button_display = False
|
||||||
else:
|
else:
|
||||||
next_workorder = sorted_workorders[position + 1]
|
next_workorder = sorted_workorders[position + 1]
|
||||||
next_state = next_workorder.state
|
# 持续获取下一个工单,直到找到一个不是返工的工单
|
||||||
if (next_state == 'ready' or (
|
while next_workorder and next_workorder.state == 'rework':
|
||||||
|
position += 1
|
||||||
|
if position + 1 < len(sorted_workorders):
|
||||||
|
next_workorder = sorted_workorders[position + 1]
|
||||||
|
else:
|
||||||
|
next_workorder = None
|
||||||
|
if next_workorder and (next_workorder.state == 'ready' or (
|
||||||
next_workorder.state == 'waiting' and next_workorder.is_subcontract)) and cur_workorder.state == 'done':
|
next_workorder.state == 'waiting' and next_workorder.is_subcontract)) and cur_workorder.state == 'done':
|
||||||
record.back_button_display = True
|
record.back_button_display = True
|
||||||
else:
|
else:
|
||||||
@@ -220,22 +227,30 @@ class ResMrpWorkOrder(models.Model):
|
|||||||
# finish_move.move_dest_ids.move_line_ids.reserved_uom_qty = 0
|
# finish_move.move_dest_ids.move_line_ids.reserved_uom_qty = 0
|
||||||
else:
|
else:
|
||||||
next_workorder = sorted_workorders[position + 1]
|
next_workorder = sorted_workorders[position + 1]
|
||||||
next_state = next_workorder.state
|
# 持续获取下一个工单,直到找到一个不是返工的工单
|
||||||
if next_state not in ['pending', 'waiting', 'ready']:
|
while next_workorder and next_workorder.state == 'rework':
|
||||||
raise UserError('下工序已经开始,无法回退')
|
position += 1
|
||||||
if next_workorder.is_subcontract:
|
if position + 1 < len(sorted_workorders):
|
||||||
next_workorder.picking_ids.write({'state': 'waiting'})
|
next_workorder = sorted_workorders[position + 1]
|
||||||
next_workorder.state = 'pending'
|
|
||||||
self.time_ids.date_end = None
|
|
||||||
cur_workorder.state = 'progress'
|
|
||||||
cur_workorder.production_id.state = 'progress'
|
|
||||||
quality_check = self.env['quality.check'].search(
|
|
||||||
[('workorder_id', '=', self.id)])
|
|
||||||
for check_order in quality_check:
|
|
||||||
if check_order.point_id.is_inspect:
|
|
||||||
check_order.quality_state = 'waiting'
|
|
||||||
else:
|
else:
|
||||||
check_order.quality_state = 'none'
|
next_workorder = None
|
||||||
|
if next_workorder:
|
||||||
|
next_state = next_workorder.state
|
||||||
|
if next_state not in ['pending', 'waiting', 'ready']:
|
||||||
|
raise UserError('下工序已经开始,无法回退')
|
||||||
|
if next_workorder.is_subcontract:
|
||||||
|
next_workorder.picking_ids.write({'state': 'waiting'})
|
||||||
|
next_workorder.state = 'pending'
|
||||||
|
self.time_ids.date_end = None
|
||||||
|
cur_workorder.state = 'progress'
|
||||||
|
cur_workorder.production_id.state = 'progress'
|
||||||
|
quality_check = self.env['quality.check'].search(
|
||||||
|
[('workorder_id', '=', self.id)])
|
||||||
|
for check_order in quality_check:
|
||||||
|
if check_order.point_id.is_inspect:
|
||||||
|
check_order.quality_state = 'waiting'
|
||||||
|
else:
|
||||||
|
check_order.quality_state = 'none'
|
||||||
|
|
||||||
def _compute_working_users(self):
|
def _compute_working_users(self):
|
||||||
super()._compute_working_users()
|
super()._compute_working_users()
|
||||||
@@ -319,6 +334,7 @@ class ResMrpWorkOrder(models.Model):
|
|||||||
|
|
||||||
tag_type = fields.Selection([("重新加工", "重新加工")], string="标签", tracking=True)
|
tag_type = fields.Selection([("重新加工", "重新加工")], string="标签", tracking=True)
|
||||||
technology_design_id = fields.Many2one('sf.technology.design')
|
technology_design_id = fields.Many2one('sf.technology.design')
|
||||||
|
cnc_worksheet_name = fields.Char('工作指令文件名', readonly=True)
|
||||||
|
|
||||||
def _compute_default_construction_period_status(self):
|
def _compute_default_construction_period_status(self):
|
||||||
need_list = ['pending', 'waiting', 'ready', 'progress', 'to be detected', 'done']
|
need_list = ['pending', 'waiting', 'ready', 'progress', 'to be detected', 'done']
|
||||||
@@ -439,15 +455,15 @@ class ResMrpWorkOrder(models.Model):
|
|||||||
action['context'] = dict(self._context, default_origin=self.name)
|
action['context'] = dict(self._context, default_origin=self.name)
|
||||||
return action
|
return action
|
||||||
|
|
||||||
@api.depends('state', 'production_id.name')
|
|
||||||
def _compute_surface_technics_purchase_ids(self):
|
def _compute_surface_technics_purchase_ids(self):
|
||||||
for order in self:
|
for order in self:
|
||||||
if order.routing_type == '表面工艺' and order.state not in ['cancel']:
|
if order.routing_type == '表面工艺' and order.state not in ['cancel']:
|
||||||
# domain = [('group_id', '=', self.production_id.procurement_group_id.id),
|
|
||||||
# ('purchase_type', '=', 'consignment'), ('state', '!=', 'cancel')]
|
|
||||||
domain = [('purchase_type', '=', 'consignment'),
|
domain = [('purchase_type', '=', 'consignment'),
|
||||||
('origin', 'like', '%' + self.production_id.name + '%'),
|
('origin', 'like', '%' + self.production_id.name + '%'),
|
||||||
('state', '!=', 'cancel')]
|
('state', '!=', 'cancel')]
|
||||||
|
# domain = [('purchase_type', '=', 'consignment'),
|
||||||
|
# ('origin', 'like', '%' + self.production_id.name + '%'),
|
||||||
|
# ('state', '!=', 'cancel')]
|
||||||
purchase = self.env['purchase.order'].search(domain)
|
purchase = self.env['purchase.order'].search(domain)
|
||||||
order.surface_technics_purchase_count = 0
|
order.surface_technics_purchase_count = 0
|
||||||
if not purchase:
|
if not purchase:
|
||||||
@@ -460,30 +476,31 @@ class ResMrpWorkOrder(models.Model):
|
|||||||
else:
|
else:
|
||||||
order.surface_technics_purchase_count = 0
|
order.surface_technics_purchase_count = 0
|
||||||
|
|
||||||
def action_view_pr_mrp_workorder(self):
|
# def action_view_pr_mrp_workorder(self):
|
||||||
"""
|
# """
|
||||||
采购请求
|
# 采购请求
|
||||||
"""
|
# """
|
||||||
self.ensure_one()
|
# self.ensure_one()
|
||||||
pr_ids = self.env['purchase.request'].sudo().search(
|
# pr_ids = self.env['purchase.request'].sudo().search(
|
||||||
[('origin', 'like', self.production_id.name), ('is_subcontract', '=', 'True'),
|
# [('origin', 'like', self.production_id.name), ('is_subcontract', '=', 'True'),
|
||||||
('state', '!=', 'rejected')])
|
# ('state', '!=', 'rejected')])
|
||||||
action = {
|
# action = {
|
||||||
'res_model': 'purchase.request',
|
# 'res_model': 'purchase.request',
|
||||||
'type': 'ir.actions.act_window',
|
# 'type': 'ir.actions.act_window',
|
||||||
}
|
# }
|
||||||
if len(pr_ids) == 1:
|
# if len(pr_ids) == 1:
|
||||||
action.update({
|
# action.update({
|
||||||
'view_mode': 'form',
|
# 'view_mode': 'form',
|
||||||
'res_id': pr_ids[0].id,
|
# 'res_id': pr_ids[0].id,
|
||||||
})
|
# })
|
||||||
else:
|
# else:
|
||||||
action.update({
|
# action.update({
|
||||||
'name': _("从 %s生成采购请求单", self.name),
|
# 'name': _("从 %s生成采购请求单", self.name),
|
||||||
'domain': [('id', 'in', pr_ids)],
|
# 'domain': [('id', 'in', pr_ids)],
|
||||||
'view_mode': 'tree,form',
|
# 'view_mode': 'tree,form',
|
||||||
})
|
# })
|
||||||
return action
|
# return action
|
||||||
|
|
||||||
def action_view_surface_technics_purchase(self):
|
def action_view_surface_technics_purchase(self):
|
||||||
self.ensure_one()
|
self.ensure_one()
|
||||||
# if self.routing_type == '表面工艺':
|
# if self.routing_type == '表面工艺':
|
||||||
@@ -512,10 +529,11 @@ class ResMrpWorkOrder(models.Model):
|
|||||||
return result
|
return result
|
||||||
|
|
||||||
def _get_surface_technics_purchase_ids(self):
|
def _get_surface_technics_purchase_ids(self):
|
||||||
domain = [('origin', 'like', '%' + self.production_id.name + '%'), ('purchase_type', '=', 'consignment')]
|
domain = [('origin', 'like', '%' + self.production_id.name + '%'), ('purchase_type', '=', 'consignment'),
|
||||||
|
('state', '!=', 'cancel')]
|
||||||
|
# domain = [('origin', 'like', '%' + self.production_id.name + '%'), ('purchase_type', '=', 'consignment')]
|
||||||
# domain = [('group_id', '=', self.production_id.procurement_group_id.id), ('purchase_type', '=', 'consignment')]
|
# domain = [('group_id', '=', self.production_id.procurement_group_id.id), ('purchase_type', '=', 'consignment')]
|
||||||
purchase_orders = self.env['purchase.order'].search(domain, order='id desc', # 按创建时间降序(最新的在前)
|
purchase_orders = self.env['purchase.order'].search(domain, order='id desc')
|
||||||
limit=1)
|
|
||||||
purchase_orders_id = self.env['purchase.order']
|
purchase_orders_id = self.env['purchase.order']
|
||||||
for po in purchase_orders:
|
for po in purchase_orders:
|
||||||
for line in po.order_line:
|
for line in po.order_line:
|
||||||
@@ -735,21 +753,25 @@ class ResMrpWorkOrder(models.Model):
|
|||||||
# self.workpiece_delivery_ids[0].write({'rfid_code': self.rfid_code})
|
# self.workpiece_delivery_ids[0].write({'rfid_code': self.rfid_code})
|
||||||
|
|
||||||
def get_plan_workorder(self, production_line):
|
def get_plan_workorder(self, production_line):
|
||||||
tomorrow = (date.today() + timedelta(days=+1)).strftime("%Y-%m-%d")
|
tomorrow = (date.today() + timedelta(days=1)).strftime("%Y-%m-%d")
|
||||||
tomorrow_start = tomorrow + ' 00:00:00'
|
tomorrow_start = f"{tomorrow} 00:00:00"
|
||||||
tomorrow_end = tomorrow + ' 23:59:59'
|
tomorrow_end = f"{tomorrow} 23:59:59"
|
||||||
logging.info('tomorrow:%s' % tomorrow)
|
logging.info('tomorrow:%s' % tomorrow)
|
||||||
sql = """
|
sql = """
|
||||||
SELECT *
|
SELECT *
|
||||||
FROM mrp_workorder
|
FROM mrp_workorder
|
||||||
WHERE state!='rework'
|
WHERE state!='rework'
|
||||||
to_char(date_planned_start::timestamp + '8 hour','YYYY-MM-DD HH:mm:SS')>= %s
|
AND (date_planned_start + interval '8 hours') >= %s::timestamp
|
||||||
AND to_char(date_planned_finished::timestamp + '8 hour','YYYY-MM-DD HH:mm:SS')<= %s
|
AND (date_planned_finished + interval '8 hours') <= %s::timestamp
|
||||||
"""
|
"""
|
||||||
params = [tomorrow_start, tomorrow_end]
|
params = [tomorrow_start, tomorrow_end]
|
||||||
if production_line:
|
if production_line:
|
||||||
|
line = self.env['sf.production.line'].search(
|
||||||
|
[('name', '=', production_line)], limit=1)
|
||||||
|
if not line:
|
||||||
|
raise ValueError(f"生产线'{production_line}'不存在")
|
||||||
sql += "AND production_line_id = %s"
|
sql += "AND production_line_id = %s"
|
||||||
params.append(production_line)
|
params.append(line.id)
|
||||||
self.env.cr.execute(sql, params)
|
self.env.cr.execute(sql, params)
|
||||||
ids = [t[0] for t in self.env.cr.fetchall()]
|
ids = [t[0] for t in self.env.cr.fetchall()]
|
||||||
return [('id', 'in', ids)]
|
return [('id', 'in', ids)]
|
||||||
@@ -1242,8 +1264,16 @@ class ResMrpWorkOrder(models.Model):
|
|||||||
}]
|
}]
|
||||||
return workorders_values_str
|
return workorders_values_str
|
||||||
|
|
||||||
|
# def check_lot_exists(self, picking_id, lot_id):
|
||||||
|
# return bool(
|
||||||
|
# picking_id.move_ids.move_line_ids.filtered(
|
||||||
|
# lambda line: line.lot_id.id == lot_id
|
||||||
|
# )
|
||||||
|
# )
|
||||||
|
|
||||||
def _process_compute_state(self):
|
def _process_compute_state(self):
|
||||||
for workorder in self:
|
sorted_workorders = sorted(self, key=lambda x: x.sequence)
|
||||||
|
for workorder in sorted_workorders:
|
||||||
# 如果工单的工序没有进行排序则跳出循环
|
# 如果工单的工序没有进行排序则跳出循环
|
||||||
if workorder.production_id.workorder_ids.filtered(lambda wk: wk.sequence == 0):
|
if workorder.production_id.workorder_ids.filtered(lambda wk: wk.sequence == 0):
|
||||||
continue
|
continue
|
||||||
@@ -1262,10 +1292,17 @@ class ResMrpWorkOrder(models.Model):
|
|||||||
workorder.state = 'pending'
|
workorder.state = 'pending'
|
||||||
continue
|
continue
|
||||||
# ================= 如果制造订单制造类型为【人工线下加工】==========================
|
# ================= 如果制造订单制造类型为【人工线下加工】==========================
|
||||||
|
# lot_id = workorder.production_id.move_raw_ids.move_line_ids.lot_id
|
||||||
|
# picking_ids = workorder.production_id.picking_ids.filtered(
|
||||||
|
# lambda wk: wk.location_id.name == '外协收料区' and wk.location_dest_id.name == '制造前')
|
||||||
|
# exists = any(
|
||||||
|
# move_line.lot_id == lot_id
|
||||||
|
# for picking in picking_ids
|
||||||
|
# for move in picking.move_ids
|
||||||
|
# for move_line in move.move_line_ids
|
||||||
|
# )
|
||||||
if (workorder.production_id.production_type == '人工线下加工'
|
if (workorder.production_id.production_type == '人工线下加工'
|
||||||
and workorder.production_id.schedule_state == '已排'
|
and workorder.production_id.schedule_state == '已排'):
|
||||||
and len(workorder.production_id.picking_ids.filtered(
|
|
||||||
lambda w: w.state not in ['done', 'cancel'])) == 0):
|
|
||||||
# and workorder.production_id.programming_state == '已编程'
|
# and workorder.production_id.programming_state == '已编程'
|
||||||
if workorder.is_subcontract is True:
|
if workorder.is_subcontract is True:
|
||||||
if workorder.production_id.state == 'rework':
|
if workorder.production_id.state == 'rework':
|
||||||
@@ -1274,6 +1311,9 @@ class ResMrpWorkOrder(models.Model):
|
|||||||
purchase_orders_id = self._get_surface_technics_purchase_ids()
|
purchase_orders_id = self._get_surface_technics_purchase_ids()
|
||||||
if purchase_orders_id.state == 'purchase':
|
if purchase_orders_id.state == 'purchase':
|
||||||
workorder.state = 'ready'
|
workorder.state = 'ready'
|
||||||
|
# picking_id = workorder.production_id.picking_ids.filtered(
|
||||||
|
# lambda wk: wk.location_id.name == '制造前' and wk.location_dest_id.name == '外协加工区')
|
||||||
|
# move_out = picking_id.move_ids
|
||||||
move_out = workorder.move_subcontract_workorder_ids[1]
|
move_out = workorder.move_subcontract_workorder_ids[1]
|
||||||
for mo in move_out:
|
for mo in move_out:
|
||||||
if mo.state != 'done':
|
if mo.state != 'done':
|
||||||
@@ -1302,13 +1342,6 @@ class ResMrpWorkOrder(models.Model):
|
|||||||
workorder.state = 'ready'
|
workorder.state = 'ready'
|
||||||
elif workorder.state != 'waiting':
|
elif workorder.state != 'waiting':
|
||||||
workorder.state = 'waiting'
|
workorder.state = 'waiting'
|
||||||
# =========== 特殊工艺工单处理 ===================
|
|
||||||
# if workorder.routing_type == '表面工艺' and workorder.is_subcontrac:
|
|
||||||
# purchase_order = self.env['purchase.order'].search(
|
|
||||||
# [('origin', 'ilike', workorder.production_id.name)])
|
|
||||||
# if purchase_order.picking_ids.filtered(lambda p: p.state in ['waiting', 'confirmed', 'assigned']):
|
|
||||||
# workorder.state = 'waiting'
|
|
||||||
# continue
|
|
||||||
if workorder.technology_design_id.routing_tag == 'special':
|
if workorder.technology_design_id.routing_tag == 'special':
|
||||||
if workorder.is_subcontract is False:
|
if workorder.is_subcontract is False:
|
||||||
workorder.state = 'ready'
|
workorder.state = 'ready'
|
||||||
@@ -1321,6 +1354,10 @@ class ResMrpWorkOrder(models.Model):
|
|||||||
if purchase_orders_id.state == 'purchase':
|
if purchase_orders_id.state == 'purchase':
|
||||||
workorder.state = 'ready'
|
workorder.state = 'ready'
|
||||||
move_out = workorder.move_subcontract_workorder_ids[1]
|
move_out = workorder.move_subcontract_workorder_ids[1]
|
||||||
|
# picking_id = workorder.production_id.picking_ids.filtered(
|
||||||
|
# lambda
|
||||||
|
# wk: wk.location_id.name == '制造前' and wk.location_dest_id.name == '外协加工区')
|
||||||
|
# move_out = picking_id.move_ids
|
||||||
for mo in move_out:
|
for mo in move_out:
|
||||||
if mo.state != 'done':
|
if mo.state != 'done':
|
||||||
mo.write({'state': 'assigned', 'production_id': False})
|
mo.write({'state': 'assigned', 'production_id': False})
|
||||||
@@ -1361,11 +1398,11 @@ class ResMrpWorkOrder(models.Model):
|
|||||||
# 判断是否有坯料的序列号信息
|
# 判断是否有坯料的序列号信息
|
||||||
boolean = False
|
boolean = False
|
||||||
if self.production_id.move_raw_ids:
|
if self.production_id.move_raw_ids:
|
||||||
if self.production_id.move_raw_ids[0].product_id.categ_type == '坯料':
|
if self.production_id.move_raw_ids[0].product_id.categ_type == '坯料' and \
|
||||||
|
self.production_id.move_raw_ids[0].product_id.tracking == 'serial':
|
||||||
if self.production_id.move_raw_ids[0].move_line_ids:
|
if self.production_id.move_raw_ids[0].move_line_ids:
|
||||||
if self.production_id.move_raw_ids[0].move_line_ids:
|
if self.production_id.move_raw_ids[0].move_line_ids[0].lot_id.name:
|
||||||
if self.production_id.move_raw_ids[0].move_line_ids[0].lot_id.name:
|
boolean = True
|
||||||
boolean = True
|
|
||||||
else:
|
else:
|
||||||
boolean = True
|
boolean = True
|
||||||
if not boolean:
|
if not boolean:
|
||||||
@@ -1395,6 +1432,10 @@ class ResMrpWorkOrder(models.Model):
|
|||||||
if self.routing_type == '表面工艺':
|
if self.routing_type == '表面工艺':
|
||||||
if self.is_subcontract is True:
|
if self.is_subcontract is True:
|
||||||
move_out = self.move_subcontract_workorder_ids[1]
|
move_out = self.move_subcontract_workorder_ids[1]
|
||||||
|
# picking_id = self.production_id.picking_ids.filtered(
|
||||||
|
# lambda wk: wk.location_id.name == '制造前' and wk.location_dest_id.name == '外协加工区')
|
||||||
|
# move_out = picking_id.move_ids
|
||||||
|
# move_out = self.move_subcontract_workorder_ids[1]
|
||||||
# move_out = self.env['stock.move'].search(
|
# move_out = self.env['stock.move'].search(
|
||||||
# [('location_id', '=', self.env['stock.location'].search(
|
# [('location_id', '=', self.env['stock.location'].search(
|
||||||
# [('barcode', 'ilike', 'WH-PREPRODUCTION')]).id),
|
# [('barcode', 'ilike', 'WH-PREPRODUCTION')]).id),
|
||||||
@@ -1560,25 +1601,17 @@ class ResMrpWorkOrder(models.Model):
|
|||||||
len(done_workorder) == len(
|
len(done_workorder) == len(
|
||||||
record.production_id.workorder_ids.filtered(lambda wo: wo.state != 'cancel'))):
|
record.production_id.workorder_ids.filtered(lambda wo: wo.state != 'cancel'))):
|
||||||
is_production_id = True
|
is_production_id = True
|
||||||
if record.routing_type in ['解除装夹'] or (
|
|
||||||
record.is_rework is True and record.routing_type in ['装夹预调']):
|
|
||||||
for workorder in record.production_id.workorder_ids:
|
|
||||||
if workorder.processing_panel == record.processing_panel:
|
|
||||||
rfid_code = workorder.rfid_code
|
|
||||||
if record.is_rework is not True:
|
|
||||||
workorder.write({'rfid_code_old': rfid_code, 'rfid_code': False})
|
|
||||||
elif workorder.routing_type != '装夹预调' and workorder.state != 'rework':
|
|
||||||
workorder.write({'rfid_code_old': False, 'rfid_code': False})
|
|
||||||
elif workorder.routing_type == '装夹预调' and workorder.state != 'rework':
|
|
||||||
workorder.write({'rfid_code_old': rfid_code, 'rfid_code': False})
|
|
||||||
self.env['stock.lot'].sudo().search([('rfid', '=', rfid_code)]).write(
|
|
||||||
{'tool_material_status': '可用'})
|
|
||||||
if workorder.rfid_code:
|
|
||||||
raise ValidationError(f'【{workorder.name}】工单解绑失败,请重新点击完成按钮!!!')
|
|
||||||
# workorder.rfid_code_old = rfid_code
|
|
||||||
# workorder.rfid_code = False
|
|
||||||
logging.info('workorder.rfid_code:%s' % workorder.rfid_code)
|
|
||||||
|
|
||||||
|
if record.routing_type in ['解除装夹']:
|
||||||
|
rfid_code = record.rfid_code
|
||||||
|
work_ids = record.production_id.workorder_ids.filtered(
|
||||||
|
lambda wo: wo.processing_panel == record.processing_panel and wo.state != 'rework')
|
||||||
|
work_ids.write({'rfid_code_old': rfid_code, 'rfid_code': False})
|
||||||
|
self.env['stock.lot'].sudo().search([('rfid', '=', rfid_code)]).write(
|
||||||
|
{'tool_material_status': '可用'})
|
||||||
|
if any(wo.rfid_code for wo in work_ids):
|
||||||
|
raise ValidationError(f'【{record.name}】工单解绑失败,请重新点击完成按钮!!!')
|
||||||
|
logging.info('work_ids.rfid_code:%s' % [wo.rfid_code for wo in work_ids])
|
||||||
if is_production_id is True:
|
if is_production_id is True:
|
||||||
logging.info('product_qty:%s' % record.production_id.product_qty)
|
logging.info('product_qty:%s' % record.production_id.product_qty)
|
||||||
for move_raw_id in record.production_id.move_raw_ids:
|
for move_raw_id in record.production_id.move_raw_ids:
|
||||||
@@ -1813,8 +1846,8 @@ class ResMrpWorkOrder(models.Model):
|
|||||||
orderby=orderby,
|
orderby=orderby,
|
||||||
lazy=lazy
|
lazy=lazy
|
||||||
)
|
)
|
||||||
|
|
||||||
model_id = fields.Char('模型id', related='production_id.model_id')
|
model_id = fields.Char('模型ID', related='production_id.model_id')
|
||||||
|
|
||||||
|
|
||||||
class CNCprocessing(models.Model):
|
class CNCprocessing(models.Model):
|
||||||
|
|||||||
@@ -10,8 +10,8 @@ from odoo.exceptions import ValidationError, UserError
|
|||||||
from odoo.modules import get_resource_path
|
from odoo.modules import get_resource_path
|
||||||
|
|
||||||
|
|
||||||
from OCC.Extend.DataExchange import read_step_file
|
# from OCC.Extend.DataExchange import read_step_file
|
||||||
from OCC.Extend.DataExchange import write_stl_file
|
# from OCC.Extend.DataExchange import write_stl_file
|
||||||
|
|
||||||
|
|
||||||
class ResProductMo(models.Model):
|
class ResProductMo(models.Model):
|
||||||
@@ -26,9 +26,12 @@ class ResProductMo(models.Model):
|
|||||||
model_file = fields.Binary('模型文件')
|
model_file = fields.Binary('模型文件')
|
||||||
categ_type = fields.Selection(string='产品的类别', related='categ_id.type', store=True)
|
categ_type = fields.Selection(string='产品的类别', related='categ_id.type', store=True)
|
||||||
model_name = fields.Char('模型名称')
|
model_name = fields.Char('模型名称')
|
||||||
|
blank_type = fields.Selection([('圆料', '圆料'), ('方料', '方料')], string='坯料分类')
|
||||||
|
blank_precision = fields.Selection([('精坯', '精坯'), ('粗坯', '粗坯')], string='坯料类型')
|
||||||
model_long = fields.Float('模型长(mm)', digits=(16, 3))
|
model_long = fields.Float('模型长(mm)', digits=(16, 3))
|
||||||
model_width = fields.Float('模型宽(mm)', digits=(16, 3))
|
model_width = fields.Float('模型宽(mm)', digits=(16, 3))
|
||||||
model_height = fields.Float('模型高(mm)', digits=(16, 3))
|
model_height = fields.Float('模型高(mm)', digits=(16, 3))
|
||||||
|
unit_number = fields.Float('单件用量', digits=(16, 3), default=1)
|
||||||
model_volume = fields.Float('模型体积(m³)')
|
model_volume = fields.Float('模型体积(m³)')
|
||||||
model_area = fields.Float('模型表面积(m²)')
|
model_area = fields.Float('模型表面积(m²)')
|
||||||
model_machining_precision = fields.Selection(selection=_get_machining_precision, string='加工精度')
|
model_machining_precision = fields.Selection(selection=_get_machining_precision, string='加工精度')
|
||||||
@@ -787,7 +790,7 @@ class ResProductMo(models.Model):
|
|||||||
glb_url = fields.Char('glb文件地址')
|
glb_url = fields.Char('glb文件地址')
|
||||||
area = fields.Float('表面积(m²)')
|
area = fields.Float('表面积(m²)')
|
||||||
auto_machining = fields.Boolean('自动化加工(模型识别)', default=False)
|
auto_machining = fields.Boolean('自动化加工(模型识别)', default=False)
|
||||||
model_id = fields.Char('模型id')
|
model_id = fields.Char('模型ID')
|
||||||
|
|
||||||
|
|
||||||
@api.depends('name')
|
@api.depends('name')
|
||||||
@@ -795,10 +798,12 @@ class ResProductMo(models.Model):
|
|||||||
for record in self:
|
for record in self:
|
||||||
if record.categ_id.name == '坯料':
|
if record.categ_id.name == '坯料':
|
||||||
product_name = ''
|
product_name = ''
|
||||||
match = re.search(r'(S\d{5}-\d)', record.name)
|
match = re.search(r'(S\d{5}-\d+)', record.name)
|
||||||
# 如果匹配成功,提取结果
|
# 如果匹配成功,提取结果
|
||||||
if match:
|
if match:
|
||||||
product_name = match.group(0)
|
product_name = match.group(0)
|
||||||
|
else:
|
||||||
|
product_name = record.name
|
||||||
sale_order_name = ''
|
sale_order_name = ''
|
||||||
match_sale = re.search(r'S(\d+)', record.name)
|
match_sale = re.search(r'S(\d+)', record.name)
|
||||||
if match_sale:
|
if match_sale:
|
||||||
@@ -890,14 +895,23 @@ class ResProductMo(models.Model):
|
|||||||
embryo_redundancy_id = item.get('embryo_redundancy')
|
embryo_redundancy_id = item.get('embryo_redundancy')
|
||||||
if not embryo_redundancy_id:
|
if not embryo_redundancy_id:
|
||||||
raise UserError('请先配置模型类型内的坯料冗余')
|
raise UserError('请先配置模型类型内的坯料冗余')
|
||||||
|
product_name = self.generate_product_name(order_id, item, i)
|
||||||
|
# 判断参数中是否包含 坯料尺寸(长、宽、高)
|
||||||
|
blank_bool = any(value is not None and value != 0 for value in (
|
||||||
|
item.get('blank_length'), item.get('blank_width'), item.get('blank_height'))) if all(
|
||||||
|
key in item for key in ('blank_length', 'blank_width', 'blank_height')) else False
|
||||||
vals = {
|
vals = {
|
||||||
'name': '%s-%s-%s' % ('P', order_id.name, i),
|
'name': product_name,
|
||||||
'model_long': self.format_float(item['model_long'] + embryo_redundancy_id.long),
|
'blank_type': item.get('blank_type'),
|
||||||
'model_width': self.format_float(item['model_width'] + embryo_redundancy_id.width),
|
'blank_precision': item.get('blank_precision'),
|
||||||
'model_height': self.format_float(item['model_height'] + embryo_redundancy_id.height),
|
'model_long': item.get('blank_length') if blank_bool else self.format_float(item['model_long'] + embryo_redundancy_id.long),
|
||||||
'model_volume': self.format_float((item['model_long'] + embryo_redundancy_id.long) * (
|
'model_width': item.get('blank_width') if blank_bool else self.format_float(item['model_width'] + embryo_redundancy_id.width),
|
||||||
item['model_width'] + embryo_redundancy_id.width) * (
|
'model_height': item.get('blank_height') if blank_bool else self.format_float(item['model_height'] + embryo_redundancy_id.height),
|
||||||
item['model_height'] + embryo_redundancy_id.height)),
|
'unit_number': item.get('unit_number'),
|
||||||
|
'model_volume': self.format_float(((item['model_long'] + embryo_redundancy_id.long) *
|
||||||
|
(item['model_width'] + embryo_redundancy_id.width) *
|
||||||
|
(item['model_height'] + embryo_redundancy_id.height))) if not blank_bool else (
|
||||||
|
item.get('blank_length') * item.get('blank_width') * item.get('blank_height')),
|
||||||
'product_model_type_id': model_type.id,
|
'product_model_type_id': model_type.id,
|
||||||
'model_processing_panel': item['processing_panel_detail'],
|
'model_processing_panel': item['processing_panel_detail'],
|
||||||
'model_machining_precision': item['model_machining_precision'],
|
'model_machining_precision': item['model_machining_precision'],
|
||||||
@@ -947,7 +961,7 @@ class ResProductMo(models.Model):
|
|||||||
self.attachment_update(item['quality_standard_name'], copy_product_id.product_tmpl_id.id,
|
self.attachment_update(item['quality_standard_name'], copy_product_id.product_tmpl_id.id,
|
||||||
'quality_standard', item['quality_standard_mimetype'])
|
'quality_standard', item['quality_standard_mimetype'])
|
||||||
return copy_product_id
|
return copy_product_id
|
||||||
|
|
||||||
def format_float(self, value):
|
def format_float(self, value):
|
||||||
# 将浮点数转换为字符串
|
# 将浮点数转换为字符串
|
||||||
value_str = str(value)
|
value_str = str(value)
|
||||||
@@ -1012,12 +1026,9 @@ class ResProductMo(models.Model):
|
|||||||
if not embryo_redundancy_id:
|
if not embryo_redundancy_id:
|
||||||
raise UserError('请先配置模型类型内的坯料冗余')
|
raise UserError('请先配置模型类型内的坯料冗余')
|
||||||
logging.info('no_bom_copy_product_supplier-vals:%s' % supplier)
|
logging.info('no_bom_copy_product_supplier-vals:%s' % supplier)
|
||||||
|
embryo_name = self.generate_embryo_name(order_id, item, materials_id, materials_type_id, embryo_redundancy_id, i)
|
||||||
vals = {
|
vals = {
|
||||||
'name': '%s-%s-%s [%s %s-%s * %s * %s]' % ('R',
|
'name': embryo_name,
|
||||||
order_id.name, i, materials_id.name, materials_type_id.name,
|
|
||||||
self.format_float(item['model_long'] + embryo_redundancy_id.long),
|
|
||||||
self.format_float(item['model_width'] + embryo_redundancy_id.width),
|
|
||||||
self.format_float(item['model_height'] + embryo_redundancy_id.height)),
|
|
||||||
'length': self.format_float(item['model_long'] + embryo_redundancy_id.long),
|
'length': self.format_float(item['model_long'] + embryo_redundancy_id.long),
|
||||||
'width': self.format_float(item['model_width'] + embryo_redundancy_id.width),
|
'width': self.format_float(item['model_width'] + embryo_redundancy_id.width),
|
||||||
'height': self.format_float(item['model_height'] + embryo_redundancy_id.height),
|
'height': self.format_float(item['model_height'] + embryo_redundancy_id.height),
|
||||||
@@ -1030,6 +1041,7 @@ class ResProductMo(models.Model):
|
|||||||
'single_manufacturing': product_id.single_manufacturing,
|
'single_manufacturing': product_id.single_manufacturing,
|
||||||
'is_bfm': True,
|
'is_bfm': True,
|
||||||
'active': True,
|
'active': True,
|
||||||
|
'tracking': finish_product.tracking, # 坯料的跟踪方式跟随成品
|
||||||
}
|
}
|
||||||
# 外协和采购生成的坯料需要根据材料型号绑定供应商
|
# 外协和采购生成的坯料需要根据材料型号绑定供应商
|
||||||
if route_type == 'subcontract' or route_type == 'purchase':
|
if route_type == 'subcontract' or route_type == 'purchase':
|
||||||
@@ -1118,7 +1130,19 @@ class ResProductMo(models.Model):
|
|||||||
|
|
||||||
# 增加产品表面积
|
# 增加产品表面积
|
||||||
|
|
||||||
|
def generate_product_name(self, order_id, item, i):
|
||||||
|
"""生成成品名称"""
|
||||||
|
product_name = '%s-%s-%s' % ('P', order_id.name, i)
|
||||||
|
return product_name
|
||||||
|
|
||||||
|
def generate_embryo_name(self, order_id, item, materials_id, materials_type_id, embryo_redundancy_id, i):
|
||||||
|
"""生成坯料名称"""
|
||||||
|
embryo_name = '%s-%s-%s [%s %s-%s * %s * %s]' % ('R',
|
||||||
|
order_id.name, i, materials_id.name, materials_type_id.name,
|
||||||
|
self.format_float(item['model_long'] + embryo_redundancy_id.long),
|
||||||
|
self.format_float(item['model_width'] + embryo_redundancy_id.width),
|
||||||
|
self.format_float(item['model_height'] + embryo_redundancy_id.height))
|
||||||
|
return embryo_name
|
||||||
|
|
||||||
class ResProductFixture(models.Model):
|
class ResProductFixture(models.Model):
|
||||||
_inherit = 'product.template'
|
_inherit = 'product.template'
|
||||||
|
|||||||
@@ -59,6 +59,86 @@ class PurchaseOrder(models.Model):
|
|||||||
production_id = self.env['mrp.production'].search([('origin', 'in', origins)])
|
production_id = self.env['mrp.production'].search([('origin', 'in', origins)])
|
||||||
purchase.production_count = len(production_id)
|
purchase.production_count = len(production_id)
|
||||||
|
|
||||||
|
# def process_replenish(self,production,total_qty):
|
||||||
|
# record = self
|
||||||
|
# bom_line_id = production.bom_id.bom_line_ids
|
||||||
|
# replenish = self.env['stock.warehouse.orderpoint'].search([
|
||||||
|
# ('product_id', '=', bom_line_id.product_id.id),
|
||||||
|
# (
|
||||||
|
# 'location_id', '=', self.env.ref('sf_stock.stock_location_outsourcing_material_receiving_area').id),
|
||||||
|
# # ('state', 'in', ['draft', 'confirmed'])
|
||||||
|
# ], limit=1)
|
||||||
|
# if not replenish:
|
||||||
|
# replenish_model = self.env['stock.warehouse.orderpoint']
|
||||||
|
# replenish = replenish_model.create({
|
||||||
|
# 'product_id': bom_line_id.product_id.id,
|
||||||
|
# 'location_id': self.env.ref(
|
||||||
|
# 'sf_stock.stock_location_outsourcing_material_receiving_area').id,
|
||||||
|
# 'route_id': self.env.ref('sf_stock.stock_route_process_outsourcing').id,
|
||||||
|
# 'group_id': record.group_id.id,
|
||||||
|
# 'qty_to_order': total_qty,
|
||||||
|
# 'origin': record.name,
|
||||||
|
# })
|
||||||
|
# else:
|
||||||
|
# replenish.write({
|
||||||
|
# 'product_id': bom_line_id.product_id.id,
|
||||||
|
# 'location_id': self.env.ref(
|
||||||
|
# 'sf_stock.stock_location_outsourcing_material_receiving_area').id,
|
||||||
|
# 'route_id': self.env.ref('sf_stock.stock_route_process_outsourcing').id,
|
||||||
|
# 'group_id': record.group_id.id,
|
||||||
|
# 'qty_to_order': total_qty + replenish.qty_to_order,
|
||||||
|
# 'origin': record.name + ',' + replenish.origin,
|
||||||
|
# })
|
||||||
|
# replenish.action_replenish()
|
||||||
|
|
||||||
|
# def outsourcing_service_replenishment(self):
|
||||||
|
# record = self
|
||||||
|
# if record.purchase_type != 'consignment':
|
||||||
|
# return
|
||||||
|
# grouped_lines = {}
|
||||||
|
# for line in record.order_line:
|
||||||
|
# if line.related_product.id not in grouped_lines:
|
||||||
|
# grouped_lines[line.related_product.id] = []
|
||||||
|
# grouped_lines[line.related_product.id].append(line)
|
||||||
|
# for product_id,lines in grouped_lines.items():
|
||||||
|
# production = self.env['mrp.production'].search([('product_id', '=', product_id)], limit=1)
|
||||||
|
# if not production:
|
||||||
|
# continue
|
||||||
|
# total_qty = sum(line.product_qty for line in lines)
|
||||||
|
# record.process_replenish(production,total_qty)
|
||||||
|
# for product_id,lines in grouped_lines.items():
|
||||||
|
# productions = self.env['mrp.production'].search([('product_id', '=', product_id)], limit=1)
|
||||||
|
# if not productions:
|
||||||
|
# continue
|
||||||
|
# # production.bom_id.bom_line_ids.product_id
|
||||||
|
# location_id = self.env['stock.location'].search([('name', '=', '制造前')])
|
||||||
|
# quants = self.env['stock.quant'].search([
|
||||||
|
# ('product_id', '=', productions.bom_id.bom_line_ids.product_id.id),
|
||||||
|
# ('location_id', '=', location_id.id)
|
||||||
|
# ])
|
||||||
|
# total_qty = sum(quants.mapped('quantity')) # 计算该位置的总库存量
|
||||||
|
# is_available = total_qty > 0
|
||||||
|
# if not is_available:
|
||||||
|
# raise UserError('请先完成坯料入库')
|
||||||
|
# for production_id in productions:
|
||||||
|
# work_ids = production_id.workorder_ids.filtered(
|
||||||
|
# lambda wk: wk.state not in ['done', 'rework', 'cancel'])
|
||||||
|
# if not work_ids:
|
||||||
|
# continue
|
||||||
|
# min_sequence_wk = min(work_ids, key=lambda wk: wk.sequence)
|
||||||
|
# if min_sequence_wk.is_subcontract:
|
||||||
|
# picking_id = production_id.picking_ids.filtered(
|
||||||
|
# lambda wk: wk.location_id.name == '制造前' and wk.location_dest_id.name == '外协加工区')
|
||||||
|
# move_out = picking_id.move_ids
|
||||||
|
# for mo in move_out:
|
||||||
|
# if mo.state != 'done':
|
||||||
|
# mo.write({'state': 'assigned', 'production_id': False})
|
||||||
|
# if not mo.move_line_ids:
|
||||||
|
# self.env['stock.move.line'].create(
|
||||||
|
# mo.get_move_line(production_id, min_sequence_wk))
|
||||||
|
# product = self.env['mrp.production'].search([('product_id', '=', product_id)], limit=1)
|
||||||
|
# match = re.search(r'(S\d{5}-\d)',product.name)
|
||||||
|
# pass
|
||||||
def button_confirm(self):
|
def button_confirm(self):
|
||||||
for record in self:
|
for record in self:
|
||||||
for line in record.order_line:
|
for line in record.order_line:
|
||||||
@@ -66,38 +146,15 @@ class PurchaseOrder(models.Model):
|
|||||||
raise UserError('请对【产品】中的【数量】进行输入')
|
raise UserError('请对【产品】中的【数量】进行输入')
|
||||||
if line.price_unit <= 0:
|
if line.price_unit <= 0:
|
||||||
raise UserError('请对【产品】中的【单价】进行输入')
|
raise UserError('请对【产品】中的【单价】进行输入')
|
||||||
if record.purchase_type == 'consignment':
|
# record.outsourcing_service_replenishment()
|
||||||
bom_line_id = record.order_line[0].purchase_request_lines.request_id.bom_id.bom_line_ids
|
|
||||||
replenish = self.env['stock.warehouse.orderpoint'].search([
|
res = super(PurchaseOrder, self).button_confirm()
|
||||||
('product_id', '=', bom_line_id.product_id.id),
|
|
||||||
(
|
|
||||||
'location_id', '=', self.env.ref('sf_stock.stock_location_outsourcing_material_receiving_area').id),
|
|
||||||
# ('state', 'in', ['draft', 'confirmed'])
|
|
||||||
], limit=1)
|
|
||||||
if not replenish:
|
|
||||||
replenish_model = self.env['stock.warehouse.orderpoint']
|
|
||||||
replenish = replenish_model.create({
|
|
||||||
'product_id': bom_line_id.product_id.id,
|
|
||||||
'location_id': self.env.ref(
|
|
||||||
'sf_stock.stock_location_outsourcing_material_receiving_area').id,
|
|
||||||
'route_id': self.env.ref('sf_stock.stock_route_process_outsourcing').id,
|
|
||||||
'group_id': record.group_id.id,
|
|
||||||
'qty_to_order': 1,
|
|
||||||
'origin': record.name,
|
|
||||||
})
|
|
||||||
else:
|
|
||||||
replenish.write({
|
|
||||||
'product_id': bom_line_id.product_id.id,
|
|
||||||
'location_id': self.env.ref(
|
|
||||||
'sf_stock.stock_location_outsourcing_material_receiving_area').id,
|
|
||||||
'route_id': self.env.ref('sf_stock.stock_route_process_outsourcing').id,
|
|
||||||
'group_id': record.group_id.id,
|
|
||||||
'qty_to_order': 1 + replenish.qty_to_order,
|
|
||||||
'origin': record.name + ',' + replenish.origin,
|
|
||||||
})
|
|
||||||
replenish.action_replenish()
|
|
||||||
|
|
||||||
return super(PurchaseOrder, self).button_confirm()
|
for line in self.order_line:
|
||||||
|
# 将产品不追踪序列号的行项目设置qty_done
|
||||||
|
if line.move_ids and line.move_ids[0].product_id.tracking == 'none':
|
||||||
|
line.move_ids[0].quantity_done = line.move_ids[0].product_qty
|
||||||
|
return res
|
||||||
|
|
||||||
|
|
||||||
origin_sale_id = fields.Many2one('sale.order', string='销售订单号', store=True, compute='_compute_origin_sale_id')
|
origin_sale_id = fields.Many2one('sale.order', string='销售订单号', store=True, compute='_compute_origin_sale_id')
|
||||||
@@ -151,10 +208,12 @@ class PurchaseOrderLine(models.Model):
|
|||||||
continue
|
continue
|
||||||
if record.product_id.categ_id.name == '坯料':
|
if record.product_id.categ_id.name == '坯料':
|
||||||
product_name = ''
|
product_name = ''
|
||||||
match = re.search(r'(S\d{5}-\d)', record.product_id.name)
|
match = re.search(r'(S\d{5}-\d+)', record.product_id.name)
|
||||||
# 如果匹配成功,提取结果
|
# 如果匹配成功,提取结果
|
||||||
if match:
|
if match:
|
||||||
product_name = match.group(0)
|
product_name = match.group(0)
|
||||||
|
else:
|
||||||
|
product_name = record.product_id.name
|
||||||
sale_order_name = ''
|
sale_order_name = ''
|
||||||
match_sale = re.search(r'S(\d+)', record.product_id.name)
|
match_sale = re.search(r'S(\d+)', record.product_id.name)
|
||||||
if match_sale:
|
if match_sale:
|
||||||
|
|||||||
@@ -1,30 +1,30 @@
|
|||||||
# -*- coding: utf-8 -*-
|
# -*- coding: utf-8 -*-
|
||||||
import base64
|
# import base64
|
||||||
import datetime
|
# import datetime
|
||||||
import logging
|
# import logging
|
||||||
import json
|
# import json
|
||||||
import os
|
# import os
|
||||||
import re
|
# import re
|
||||||
import traceback
|
# import traceback
|
||||||
from operator import itemgetter
|
# from operator import itemgetter
|
||||||
|
|
||||||
import requests
|
# import requests
|
||||||
from itertools import groupby
|
# from itertools import groupby
|
||||||
from collections import defaultdict, namedtuple
|
# from collections import defaultdict, namedtuple
|
||||||
|
|
||||||
from odoo import api, fields, models, SUPERUSER_ID, _
|
# from odoo import api, fields, models, SUPERUSER_ID, _
|
||||||
from odoo.exceptions import UserError, ValidationError
|
# from odoo.exceptions import UserError, ValidationError
|
||||||
from odoo.tools import float_compare, float_round, float_is_zero, format_datetime
|
# from odoo.tools import float_compare, float_round, float_is_zero, format_datetime
|
||||||
|
|
||||||
|
|
||||||
class PurchaseRequestLine(models.Model):
|
# class PurchaseRequestLine(models.Model):
|
||||||
_inherit = 'purchase.request'
|
# _inherit = 'purchase.request'
|
||||||
is_subcontract = fields.Boolean(string='是否外协',default=False)
|
# is_subcontract = fields.Boolean(string='是否外协',default=False)
|
||||||
class PurchaseRequestLine(models.Model):
|
# class PurchaseRequestLine(models.Model):
|
||||||
_inherit = 'purchase.request.line'
|
# _inherit = 'purchase.request.line'
|
||||||
is_subcontract = fields.Boolean(string='是否外协')
|
# is_subcontract = fields.Boolean(string='是否外协')
|
||||||
|
|
||||||
|
|
||||||
class PurchaseRequest(models.Model):
|
# class PurchaseRequest(models.Model):
|
||||||
_inherit = 'purchase.request'
|
# _inherit = 'purchase.request'
|
||||||
bom_id = fields.Many2one('mrp.bom')
|
# bom_id = fields.Many2one('mrp.bom')
|
||||||
|
|||||||
@@ -58,7 +58,8 @@ class SaleOrder(models.Model):
|
|||||||
# 复制成品模板上的属性
|
# 复制成品模板上的属性
|
||||||
line.product_id.product_tmpl_id.copy_template(product_template_id)
|
line.product_id.product_tmpl_id.copy_template(product_template_id)
|
||||||
# 将模板上的single_manufacturing属性复制到成品上
|
# 将模板上的single_manufacturing属性复制到成品上
|
||||||
line.product_id.single_manufacturing = product_template_id.single_manufacturing
|
# line.product_id.single_manufacturing = product_template_id.single_manufacturing
|
||||||
|
# line.product_id.tracking = product_template_id.tracking
|
||||||
|
|
||||||
order_id = self
|
order_id = self
|
||||||
product = line.product_id
|
product = line.product_id
|
||||||
@@ -73,14 +74,17 @@ class SaleOrder(models.Model):
|
|||||||
'blank_area': product.model_area,
|
'blank_area': product.model_area,
|
||||||
'price': product.list_price,
|
'price': product.list_price,
|
||||||
'embryo_redundancy_id': line.embryo_redundancy_id,
|
'embryo_redundancy_id': line.embryo_redundancy_id,
|
||||||
|
'model_id': line.model_id
|
||||||
}
|
}
|
||||||
product_name = ''
|
product_name = ''
|
||||||
match = re.search(r'(S\d{5}-\d)', product.name)
|
match = re.search(r'(S\d{5}-\d+)', product.name)
|
||||||
|
product_seria = 0
|
||||||
# 如果匹配成功,提取结果
|
# 如果匹配成功,提取结果
|
||||||
if match:
|
if match:
|
||||||
product_name = match.group(0)
|
product_name = match.group(0)
|
||||||
# 获取成品名结尾-n的n
|
# 获取成品名结尾-n的n
|
||||||
product_seria = int(product_name.split('-')[-1])
|
product_seria = int(product_name.split('-')[-1])
|
||||||
|
|
||||||
# 成品供货方式为采购则不生成bom
|
# 成品供货方式为采购则不生成bom
|
||||||
if line.supply_method != 'purchase':
|
if line.supply_method != 'purchase':
|
||||||
bom_data = self.env['mrp.bom'].with_user(self.env.ref("base.user_admin")).get_bom(product)
|
bom_data = self.env['mrp.bom'].with_user(self.env.ref("base.user_admin")).get_bom(product)
|
||||||
@@ -190,6 +194,40 @@ class SaleOrder(models.Model):
|
|||||||
'res_id': wizard.id,
|
'res_id': wizard.id,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
def write_sale_documents(self, kw):
|
||||||
|
"""BFM 内部下单 内容补充 """
|
||||||
|
val = {}
|
||||||
|
if kw.get('contract_file_name') and kw.get('contract_file'):
|
||||||
|
document_id = self.create_sale_documents(kw.get('contract_file_name'), kw.get('contract_file'))
|
||||||
|
val.update({'contract_document_id': document_id.id})
|
||||||
|
if kw.get('contract_code') or kw.get('contract_date'):
|
||||||
|
val.update({'contract_code': kw.get('contract_code'), 'contract_date': kw.get('contract_date')})
|
||||||
|
if kw.get('customer_name'):
|
||||||
|
val.update({'customer_name': kw.get('customer_name')})
|
||||||
|
self.write(val)
|
||||||
|
|
||||||
|
def create_sale_documents(self, contract_file_name, contract_file):
|
||||||
|
# 创建ir.attachment记录
|
||||||
|
attachment = self.env['ir.attachment'].sudo().create({
|
||||||
|
'name': contract_file_name,
|
||||||
|
'type': 'binary',
|
||||||
|
'datas': contract_file,
|
||||||
|
'res_model': 'sale.order',
|
||||||
|
})
|
||||||
|
|
||||||
|
# 获取默认的文档文件夹
|
||||||
|
workspace = self.env.ref('sf_sale.documents_sales_contracts_folder_1').id
|
||||||
|
|
||||||
|
# 创建 documents.document 记录
|
||||||
|
document = self.env['documents.document'].sudo().create({
|
||||||
|
'name': contract_file_name,
|
||||||
|
'attachment_id': attachment.id,
|
||||||
|
'folder_id': workspace,
|
||||||
|
'res_model': 'sale.order',
|
||||||
|
'res_id': self.id,
|
||||||
|
})
|
||||||
|
|
||||||
|
return document
|
||||||
|
|
||||||
class SaleOrderLine(models.Model):
|
class SaleOrderLine(models.Model):
|
||||||
_inherit = 'sale.order.line'
|
_inherit = 'sale.order.line'
|
||||||
|
|||||||
@@ -2,79 +2,79 @@
|
|||||||
import logging
|
import logging
|
||||||
from odoo import fields, models, api
|
from odoo import fields, models, api
|
||||||
from odoo.exceptions import UserError, ValidationError
|
from odoo.exceptions import UserError, ValidationError
|
||||||
from odoo.tools import str2bool
|
# from odoo.tools import str2bool
|
||||||
|
|
||||||
|
|
||||||
class SfProductionProcessParameter(models.Model):
|
class SfProductionProcessParameter(models.Model):
|
||||||
_inherit = 'sf.production.process.parameter'
|
_inherit = 'sf.production.process.parameter'
|
||||||
service_products = fields.Many2one(
|
# service_products = fields.Many2one(
|
||||||
'product.template',
|
# 'product.template',
|
||||||
string='外协服务产品',compute='_compute_service_products',inverse='_inverse_service_products',
|
# string='外协服务产品',compute='_compute_service_products',inverse='_inverse_service_products',
|
||||||
store=True
|
# store=True
|
||||||
)
|
# )
|
||||||
outsourced_service_products = fields.One2many(
|
# outsourced_service_products = fields.One2many(
|
||||||
'product.template', # 另一个模型的名称
|
# 'product.template', # 另一个模型的名称
|
||||||
'server_product_process_parameters_id', # 对应的 Many2one 字段名称
|
# 'server_product_process_parameters_id', # 对应的 Many2one 字段名称
|
||||||
string='外协服务产品'
|
# string='外协服务产品'
|
||||||
)
|
# )
|
||||||
is_product_button = fields.Boolean(compute='_compute_is_product_button',default=False)
|
# is_product_button = fields.Boolean(compute='_compute_is_product_button',default=False)
|
||||||
is_delete_button = fields.Boolean(compute='_compute_is_delete_button', default=False)
|
# is_delete_button = fields.Boolean(compute='_compute_is_delete_button', default=False)
|
||||||
routing_id = fields.Many2one('mrp.routing.workcenter', string="工序")
|
# routing_id = fields.Many2one('mrp.routing.workcenter', string="工序")
|
||||||
|
|
||||||
@api.depends('outsourced_service_products')
|
# @api.depends('outsourced_service_products')
|
||||||
def _compute_service_products(self):
|
# def _compute_service_products(self):
|
||||||
for record in self:
|
# for record in self:
|
||||||
# 假设取第一条作为主明细
|
# # 假设取第一条作为主明细
|
||||||
record.service_products = record.outsourced_service_products.id if record.outsourced_service_products else False
|
# record.service_products = record.outsourced_service_products.ids if record.outsourced_service_products else False
|
||||||
|
|
||||||
def _inverse_service_products(self):
|
# def _inverse_service_products(self):
|
||||||
for record in self:
|
# for record in self:
|
||||||
if record.service_products:
|
# if record.service_products:
|
||||||
# 确保关联关系正确
|
# # 确保关联关系正确
|
||||||
record.outsourced_service_products = record.service_products.ids if record.service_products else False
|
# record.outsourced_service_products = record.service_products.ids if record.service_products else False
|
||||||
else:
|
# else:
|
||||||
record.outsourced_service_products = False
|
# record.outsourced_service_products = False
|
||||||
def name_get(self):
|
# def name_get(self):
|
||||||
result = []
|
# result = []
|
||||||
for record in self:
|
# for record in self:
|
||||||
name = f"{record.process_id.name} - {record.name}" # 自定义显示格式
|
# name = f"{record.process_id.name} - {record.name}" # 自定义显示格式
|
||||||
result.append((record.id, name))
|
# result.append((record.id, name))
|
||||||
return result
|
# return result
|
||||||
@api.constrains('outsourced_service_products')
|
# @api.constrains('outsourced_service_products')
|
||||||
def _validate_partner_limit(self):
|
# def _validate_partner_limit(self):
|
||||||
for record in self:
|
# for record in self:
|
||||||
if len(record.outsourced_service_products) > 1:
|
# if len(record.outsourced_service_products) > 1:
|
||||||
raise ValidationError("工艺参数不能与多个产品关联")
|
# raise ValidationError("工艺参数不能与多个产品关联")
|
||||||
|
|
||||||
@api.onchange('outsourced_service_products')
|
# @api.onchange('outsourced_service_products')
|
||||||
def _onchange_validate_partner_limit(self):
|
# def _onchange_validate_partner_limit(self):
|
||||||
for record in self:
|
# for record in self:
|
||||||
if len(record.outsourced_service_products) > 1:
|
# if len(record.outsourced_service_products) > 1:
|
||||||
raise ValidationError("工艺参数不能与多个产品关联")
|
# raise ValidationError("工艺参数不能与多个产品关联")
|
||||||
@api.depends('outsourced_service_products')
|
# @api.depends('outsourced_service_products')
|
||||||
def _compute_is_product_button(self):
|
# def _compute_is_product_button(self):
|
||||||
for record in self:
|
# for record in self:
|
||||||
if record.outsourced_service_products:
|
# if record.outsourced_service_products:
|
||||||
record.is_product_button = True
|
# record.is_product_button = True
|
||||||
else:
|
# else:
|
||||||
record.is_product_button = False
|
# record.is_product_button = False
|
||||||
|
|
||||||
def has_wksp_prefix(self):
|
# def has_wksp_prefix(self):
|
||||||
"""
|
# """
|
||||||
判断字符串是否以WKSP开头(不区分大小写)
|
# 判断字符串是否以WKSP开头(不区分大小写)
|
||||||
:param text: 要检查的字符串
|
# :param text: 要检查的字符串
|
||||||
:return: True/False
|
# :return: True/False
|
||||||
"""
|
# """
|
||||||
return self.code.upper().startswith('101'+self.routing_id.code)
|
# return self.code.upper().startswith('101'+self.routing_id.code)
|
||||||
@api.depends('outsourced_service_products','code')
|
# @api.depends('outsourced_service_products','code')
|
||||||
def _compute_is_delete_button(self):
|
# def _compute_is_delete_button(self):
|
||||||
for record in self:
|
# for record in self:
|
||||||
if record.outsourced_service_products and record.has_wksp_prefix():
|
# if record.outsourced_service_products and record.has_wksp_prefix():
|
||||||
record.is_delete_button = False
|
# record.is_delete_button = False
|
||||||
elif record.outsourced_service_products:
|
# elif record.outsourced_service_products:
|
||||||
record.is_delete_button = True
|
# record.is_delete_button = True
|
||||||
else:
|
# else:
|
||||||
record.is_delete_button = True
|
# record.is_delete_button = True
|
||||||
@api.model
|
@api.model
|
||||||
def _name_search(self, name, args=None, operator='ilike', limit=100, name_get_uid=None):
|
def _name_search(self, name, args=None, operator='ilike', limit=100, name_get_uid=None):
|
||||||
if self._context.get('route_id'):
|
if self._context.get('route_id'):
|
||||||
@@ -90,19 +90,19 @@ class SfProductionProcessParameter(models.Model):
|
|||||||
return self._search(domain, limit=limit, access_rights_uid=name_get_uid)
|
return self._search(domain, limit=limit, access_rights_uid=name_get_uid)
|
||||||
return super()._name_search(name, args, operator, limit, name_get_uid)
|
return super()._name_search(name, args, operator, limit, name_get_uid)
|
||||||
|
|
||||||
def action_create_service_product(self):
|
# def action_create_service_product(self):
|
||||||
if self.id: # 如果是已存在的记录
|
# if self.id: # 如果是已存在的记录
|
||||||
self.write({}) # 空写入会触发保存
|
# self.write({}) # 空写入会触发保存
|
||||||
else: # 如果是新记录
|
# else: # 如果是新记录
|
||||||
self = self.create(self._convert_to_write(self.read()[0]))
|
# self = self.create(self._convert_to_write(self.read()[0]))
|
||||||
return {
|
# return {
|
||||||
'type': 'ir.actions.act_window',
|
# 'type': 'ir.actions.act_window',
|
||||||
'name': '向导名称',
|
# 'name': '向导名称',
|
||||||
'res_model': 'product.creation.wizard',
|
# 'res_model': 'product.creation.wizard',
|
||||||
'view_mode': 'form',
|
# 'view_mode': 'form',
|
||||||
'target': 'new',
|
# 'target': 'new',
|
||||||
'context': {'default_process_parameter_id': self.id}, # 传递当前记录ID
|
# 'context': {'default_process_parameter_id': self.id}, # 传递当前记录ID
|
||||||
}
|
# }
|
||||||
#
|
#
|
||||||
# return {
|
# return {
|
||||||
# 'name': '创建服务产品',
|
# 'name': '创建服务产品',
|
||||||
@@ -116,6 +116,6 @@ class SfProductionProcessParameter(models.Model):
|
|||||||
# },
|
# },
|
||||||
# }
|
# }
|
||||||
|
|
||||||
def action_hide_service_products(self):
|
# def action_hide_service_products(self):
|
||||||
# self.outsourced_service_products.active = False
|
# # self.outsourced_service_products.active = False
|
||||||
self.active = False
|
# self.active = False
|
||||||
|
|||||||
@@ -23,7 +23,6 @@ class sf_technology_design(models.Model):
|
|||||||
# def _compute_group_uniq_id(self):
|
# def _compute_group_uniq_id(self):
|
||||||
# for record in self:
|
# for record in self:
|
||||||
|
|
||||||
|
|
||||||
def json_technology_design_str(self, k, route, i, process_parameter):
|
def json_technology_design_str(self, k, route, i, process_parameter):
|
||||||
workorders_values_str = [0, '', {
|
workorders_values_str = [0, '', {
|
||||||
'route_id': route.id if route.routing_type in ['表面工艺'] else route.route_workcenter_id.id,
|
'route_id': route.id if route.routing_type in ['表面工艺'] else route.route_workcenter_id.id,
|
||||||
@@ -36,11 +35,19 @@ class sf_technology_design(models.Model):
|
|||||||
return workorders_values_str
|
return workorders_values_str
|
||||||
|
|
||||||
def write(self, vals):
|
def write(self, vals):
|
||||||
return super(sf_technology_design, self).write(vals)
|
res = super(sf_technology_design, self).write(vals)
|
||||||
|
if 'group_uniq_id' in vals or 'process_parameters_id' in vals or 'active' in vals:
|
||||||
|
if self.production_id:
|
||||||
|
process_parameters_id = self.production_id.technology_design_ids.mapped('process_parameters_id')
|
||||||
|
if process_parameters_id.ids:
|
||||||
|
self.production_id.product_id.model_process_parameters_ids = process_parameters_id.ids
|
||||||
|
else:
|
||||||
|
self.production_id.product_id.model_process_parameters_ids = None
|
||||||
|
return res
|
||||||
|
|
||||||
def unlink_technology_design(self):
|
def unlink_technology_design(self):
|
||||||
self.active = False
|
self.active = False
|
||||||
|
|
||||||
|
|
||||||
@api.model_create_multi
|
@api.model_create_multi
|
||||||
def create(self, vals_list):
|
def create(self, vals_list):
|
||||||
for vals in vals_list:
|
for vals in vals_list:
|
||||||
@@ -48,10 +55,12 @@ class sf_technology_design(models.Model):
|
|||||||
raise ValidationError(_("工序不能为空"))
|
raise ValidationError(_("工序不能为空"))
|
||||||
result = super(sf_technology_design, self).create(vals_list)
|
result = super(sf_technology_design, self).create(vals_list)
|
||||||
for res in result:
|
for res in result:
|
||||||
record = self.search([('production_id', '=', res.production_id.id), ('active', 'in', [True, False])], order='group_uniq_id desc', limit=1)
|
record = self.search([('production_id', '=', res.production_id.id), ('active', 'in', [True, False])],
|
||||||
res.group_uniq_id=record.group_uniq_id + 1
|
order='group_uniq_id desc', limit=1)
|
||||||
|
res.group_uniq_id = record.group_uniq_id + 1
|
||||||
return result
|
return result
|
||||||
def get_duplicates_with_inactive(self,technology_designs):
|
|
||||||
|
def get_duplicates_with_inactive(self, technology_designs):
|
||||||
# 统计每个 'sequence' 出现的次数
|
# 统计每个 'sequence' 出现的次数
|
||||||
sequence_count = Counter(technology_design.sequence for technology_design in technology_designs)
|
sequence_count = Counter(technology_design.sequence for technology_design in technology_designs)
|
||||||
|
|
||||||
@@ -62,6 +71,7 @@ class sf_technology_design(models.Model):
|
|||||||
]
|
]
|
||||||
|
|
||||||
return result
|
return result
|
||||||
|
|
||||||
# def rearrange_numbering(self,self_technology_designs):
|
# def rearrange_numbering(self,self_technology_designs):
|
||||||
# inactive_designs = self.get_duplicates_with_inactive(self_technology_designs)
|
# inactive_designs = self.get_duplicates_with_inactive(self_technology_designs)
|
||||||
# if inactive_designs:
|
# if inactive_designs:
|
||||||
@@ -75,7 +85,7 @@ class sf_technology_design(models.Model):
|
|||||||
|
|
||||||
def get_technology_design(self):
|
def get_technology_design(self):
|
||||||
return {
|
return {
|
||||||
'sequence':self.sequence,
|
'sequence': self.sequence,
|
||||||
'route_id': self.route_id.id,
|
'route_id': self.route_id.id,
|
||||||
'process_parameters_id': self.process_parameters_id.id,
|
'process_parameters_id': self.process_parameters_id.id,
|
||||||
'panel': self.panel,
|
'panel': self.panel,
|
||||||
@@ -83,17 +93,19 @@ class sf_technology_design(models.Model):
|
|||||||
'time_cycle_manual': self.time_cycle_manual,
|
'time_cycle_manual': self.time_cycle_manual,
|
||||||
'is_auto': self.is_auto,
|
'is_auto': self.is_auto,
|
||||||
'active': self.active,
|
'active': self.active,
|
||||||
'group_uniq_id':self.group_uniq_id,
|
'group_uniq_id': self.group_uniq_id,
|
||||||
}
|
}
|
||||||
def sync_technology_designs(self,production_technology_designs, self_technology_designs):
|
|
||||||
|
def sync_technology_designs(self, production_technology_designs, self_technology_designs):
|
||||||
production_id = production_technology_designs[0].production_id.id
|
production_id = production_technology_designs[0].production_id.id
|
||||||
self_technology_design_dict = {item.group_uniq_id:item for item in self_technology_designs}
|
self_technology_design_dict = {item.group_uniq_id: item for item in self_technology_designs}
|
||||||
production_technology_designs_dict = {item.group_uniq_id:item for item in production_technology_designs}
|
production_technology_designs_dict = {item.group_uniq_id: item for item in production_technology_designs}
|
||||||
for technology_design in production_technology_designs:
|
for technology_design in production_technology_designs:
|
||||||
if not self_technology_design_dict.get(technology_design.group_uniq_id):
|
if not self_technology_design_dict.get(technology_design.group_uniq_id):
|
||||||
technology_design.write({'production_id': False})
|
technology_design.write({'production_id': False})
|
||||||
else:
|
else:
|
||||||
technology_design.write(self_technology_design_dict.get(technology_design.group_uniq_id).get_technology_design())
|
technology_design.write(
|
||||||
|
self_technology_design_dict.get(technology_design.group_uniq_id).get_technology_design())
|
||||||
for technology_design in self_technology_designs:
|
for technology_design in self_technology_designs:
|
||||||
if not production_technology_designs_dict.get(technology_design.group_uniq_id):
|
if not production_technology_designs_dict.get(technology_design.group_uniq_id):
|
||||||
technology_design = technology_design.get_technology_design()
|
technology_design = technology_design.get_technology_design()
|
||||||
@@ -101,9 +113,8 @@ class sf_technology_design(models.Model):
|
|||||||
technology_design.pop('group_uniq_id')
|
technology_design.pop('group_uniq_id')
|
||||||
self.env['sf.technology.design'].create(technology_design)
|
self.env['sf.technology.design'].create(technology_design)
|
||||||
|
|
||||||
|
def unified_procedure_multiple_work_orders(self, self_technology_designs, production_item):
|
||||||
|
|
||||||
def unified_procedure_multiple_work_orders(self,self_technology_designs,production_item):
|
|
||||||
technology_designs = self.env['sf.technology.design'].sudo().search(
|
technology_designs = self.env['sf.technology.design'].sudo().search(
|
||||||
[('production_id', '=', production_item.id), ('active', 'in', [True, False])])
|
[('production_id', '=', production_item.id), ('active', 'in', [True, False])])
|
||||||
self.sync_technology_designs(self_technology_designs=self_technology_designs,production_technology_designs=technology_designs)
|
self.sync_technology_designs(self_technology_designs=self_technology_designs,
|
||||||
|
production_technology_designs=technology_designs)
|
||||||
|
|||||||
@@ -564,6 +564,13 @@ class StockPicking(models.Model):
|
|||||||
|
|
||||||
part_numbers = fields.Char(string="零件图号", compute='_compute_part_info', store=True, index=True)
|
part_numbers = fields.Char(string="零件图号", compute='_compute_part_info', store=True, index=True)
|
||||||
part_names = fields.Char(string="零件名称", compute='_compute_part_info', store=True, index=True)
|
part_names = fields.Char(string="零件名称", compute='_compute_part_info', store=True, index=True)
|
||||||
|
model_id = fields.Char('模型ID', compute='_compute_model_id', store=True, index=True)
|
||||||
|
|
||||||
|
@api.depends('move_ids_without_package.model_id')
|
||||||
|
def _compute_model_id(self):
|
||||||
|
for picking in self:
|
||||||
|
model_id = picking.move_ids_without_package.mapped('model_id')
|
||||||
|
picking.model_id = ','.join(filter(None, model_id))
|
||||||
|
|
||||||
@api.depends('move_ids_without_package.part_number', 'move_ids_without_package.part_name')
|
@api.depends('move_ids_without_package.part_number', 'move_ids_without_package.part_name')
|
||||||
def _compute_part_info(self):
|
def _compute_part_info(self):
|
||||||
@@ -630,7 +637,65 @@ class StockPicking(models.Model):
|
|||||||
if lot_ids:
|
if lot_ids:
|
||||||
move.action_clear_lines_show_details()
|
move.action_clear_lines_show_details()
|
||||||
move.action_show_details()
|
move.action_show_details()
|
||||||
|
# 先进行设置数量
|
||||||
|
self.action_set_quantities_to_reservation()
|
||||||
res = super().button_validate()
|
res = super().button_validate()
|
||||||
|
# lot_ids = None
|
||||||
|
# product_ids = self.move_ids.mapped('product_id')
|
||||||
|
# if not self.move_ids[0].product_id.single_manufacturing and self.move_ids[0].product_id.tracking == 'none':
|
||||||
|
# lot_ids = self.move_ids.move_line_ids.mapped('lot_id')
|
||||||
|
# production_ids = self.sale_order_id.mrp_production_ids if self.sale_order_id else self.env['mrp.production']
|
||||||
|
# if res and self.location_id.name == '外协收料区' and self.location_dest_id.name == '制造前':
|
||||||
|
# # 如果是最后一张外协入库单,则设置库存位置的预留数量
|
||||||
|
# for production_id in production_ids:
|
||||||
|
# if lot_ids:
|
||||||
|
# lot_id = production_id.move_raw_ids.move_line_ids.lot_id
|
||||||
|
# # picking_ids = production_id.picking_ids.filtered(
|
||||||
|
# # lambda wk: wk.location_id.name == '外协收料区' and wk.location_dest_id.name == '制造前')
|
||||||
|
# if lot_id in lot_ids:
|
||||||
|
# workorder_id = production_id.workorder_ids.filtered(
|
||||||
|
# lambda a: a.state == 'progress' and a.is_subcontract)
|
||||||
|
# if not workorder_id:
|
||||||
|
# continue
|
||||||
|
# workorder_id.button_finish()
|
||||||
|
# else:
|
||||||
|
# workorder_id = production_id.workorder_ids.filtered(lambda a: a.state == 'progress' and a.is_subcontract)
|
||||||
|
# if not workorder_id:
|
||||||
|
# continue
|
||||||
|
# workorder_id.button_finish()
|
||||||
|
# # lot_id = workorder.production_id.move_raw_ids.move_line_ids.lot_id
|
||||||
|
# # picking_ids = workorder.production_id.picking_ids.filtered(
|
||||||
|
# # lambda wk: wk.location_id.name == '外协收料区' and wk.location_dest_id.name == '制造前')
|
||||||
|
|
||||||
|
# # if move_in:
|
||||||
|
# # workorder = move_in.subcontract_workorder_id
|
||||||
|
# # workorders = workorder.production_id.workorder_ids
|
||||||
|
# # subcontract_workorders = workorders.filtered(
|
||||||
|
# # lambda wo: wo.is_subcontract == True and wo.state != 'cancel').sorted('sequence')
|
||||||
|
# # # if workorder == subcontract_workorders[-1]:
|
||||||
|
# # # self.env['stock.quant']._update_reserved_quantity(
|
||||||
|
# # # move_in.product_id, move_in.location_dest_id, move_in.product_uom_qty,
|
||||||
|
# # # lot_id=move_in.move_line_ids.lot_id,
|
||||||
|
# # # package_id=False, owner_id=False, strict=False
|
||||||
|
# # # )
|
||||||
|
# # workorder.button_finish()
|
||||||
|
# if res and self.location_id.name == '制造前' and self.location_dest_id.name == '外协加工区':
|
||||||
|
# for production_id in production_ids:
|
||||||
|
# if lot_ids:
|
||||||
|
# lot_id = production_id.move_raw_ids.move_line_ids.lot_id
|
||||||
|
# # picking_ids = production_id.picking_ids.filtered(
|
||||||
|
# # lambda wk: wk.location_id.name == '外协收料区' and wk.location_dest_id.name == '制造前')
|
||||||
|
# if lot_id in lot_ids:
|
||||||
|
# workorder_id = production_id.workorder_ids.filtered(
|
||||||
|
# lambda a: a.state == 'progress' and a.is_subcontract)
|
||||||
|
# if not workorder_id:
|
||||||
|
# continue
|
||||||
|
# workorder_id.button_finish()
|
||||||
|
# else:
|
||||||
|
# workorder_id = production_id.workorder_ids.filtered(lambda a: a.state == 'ready' and a.is_subcontract)
|
||||||
|
# if not workorder_id:
|
||||||
|
# continue
|
||||||
|
# workorder_id.button_start()
|
||||||
picking_type_in = self.env.ref('sf_manufacturing.outcontract_picking_in').id
|
picking_type_in = self.env.ref('sf_manufacturing.outcontract_picking_in').id
|
||||||
if res is True and self.picking_type_id.id == picking_type_in:
|
if res is True and self.picking_type_id.id == picking_type_in:
|
||||||
# 如果是最后一张外协入库单,则设置库存位置的预留数量
|
# 如果是最后一张外协入库单,则设置库存位置的预留数量
|
||||||
@@ -675,6 +740,7 @@ class StockPicking(models.Model):
|
|||||||
|
|
||||||
# 创建 外协出库入单
|
# 创建 外协出库入单
|
||||||
def create_outcontract_picking(self, workorders, item, sorted_workorders):
|
def create_outcontract_picking(self, workorders, item, sorted_workorders):
|
||||||
|
production = workorders[0].production_id
|
||||||
for workorder in workorders:
|
for workorder in workorders:
|
||||||
if workorder.move_subcontract_workorder_ids:
|
if workorder.move_subcontract_workorder_ids:
|
||||||
workorder.move_subcontract_workorder_ids.write({'state': 'cancel'})
|
workorder.move_subcontract_workorder_ids.write({'state': 'cancel'})
|
||||||
@@ -706,7 +772,7 @@ class StockPicking(models.Model):
|
|||||||
})
|
})
|
||||||
moves_in = self.env['stock.move'].sudo().with_context(context).create(
|
moves_in = self.env['stock.move'].sudo().with_context(context).create(
|
||||||
self.env['stock.move']._get_stock_move_values_Res(item, outcontract_picking_type_in,
|
self.env['stock.move']._get_stock_move_values_Res(item, outcontract_picking_type_in,
|
||||||
procurement_group_id.id, move_dest_id))
|
procurement_group_id.id, move_dest_id, production.product_uom_qty))
|
||||||
picking_in = self.create(
|
picking_in = self.create(
|
||||||
moves_in._get_new_picking_values_Res(item, workorder, 'WH/OCIN/'))
|
moves_in._get_new_picking_values_Res(item, workorder, 'WH/OCIN/'))
|
||||||
# pick_ids.append(picking_in.id)
|
# pick_ids.append(picking_in.id)
|
||||||
@@ -716,7 +782,7 @@ class StockPicking(models.Model):
|
|||||||
# self.env.context.get('default_production_id')
|
# self.env.context.get('default_production_id')
|
||||||
moves_out = self.env['stock.move'].sudo().with_context(context).create(
|
moves_out = self.env['stock.move'].sudo().with_context(context).create(
|
||||||
self.env['stock.move']._get_stock_move_values_Res(item, outcontract_picking_type_out,
|
self.env['stock.move']._get_stock_move_values_Res(item, outcontract_picking_type_out,
|
||||||
procurement_group_id.id, moves_in.id))
|
procurement_group_id.id, moves_in.id, production.product_uom_qty))
|
||||||
workorder.write({'move_subcontract_workorder_ids': [(6, 0, [moves_in.id, moves_out.id])]})
|
workorder.write({'move_subcontract_workorder_ids': [(6, 0, [moves_in.id, moves_out.id])]})
|
||||||
picking_out = self.create(
|
picking_out = self.create(
|
||||||
moves_out._get_new_picking_values_Res(item, workorder, 'WH/OCOUT/'))
|
moves_out._get_new_picking_values_Res(item, workorder, 'WH/OCOUT/'))
|
||||||
@@ -782,6 +848,7 @@ class ReStockMove(models.Model):
|
|||||||
materiel_height = fields.Float(string='物料高度', digits=(16, 4))
|
materiel_height = fields.Float(string='物料高度', digits=(16, 4))
|
||||||
part_number = fields.Char(string='零件图号', compute='_compute_part_info', store=True)
|
part_number = fields.Char(string='零件图号', compute='_compute_part_info', store=True)
|
||||||
part_name = fields.Char(string='零件名称', compute='_compute_part_info', store=True)
|
part_name = fields.Char(string='零件名称', compute='_compute_part_info', store=True)
|
||||||
|
model_id = fields.Char('模型ID', related='product_id.model_id')
|
||||||
|
|
||||||
@api.depends('product_id')
|
@api.depends('product_id')
|
||||||
def _compute_part_info(self):
|
def _compute_part_info(self):
|
||||||
@@ -792,10 +859,12 @@ class ReStockMove(models.Model):
|
|||||||
move.part_name = move.product_id.part_name
|
move.part_name = move.product_id.part_name
|
||||||
elif move.product_id.categ_id.type == '坯料':
|
elif move.product_id.categ_id.type == '坯料':
|
||||||
product_name = ''
|
product_name = ''
|
||||||
match = re.search(r'(S\d{5}-\d)', move.product_id.name)
|
match = re.search(r'(S\d{5}-\d+)', move.product_id.name)
|
||||||
# 如果匹配成功,提取结果
|
# 如果匹配成功,提取结果
|
||||||
if match:
|
if match:
|
||||||
product_name = match.group(0)
|
product_name = match.group(0)
|
||||||
|
else:
|
||||||
|
product_name = move.product_id.name
|
||||||
if move.picking_id.sale_order_id:
|
if move.picking_id.sale_order_id:
|
||||||
sale_order = move.picking_id.sale_order_id
|
sale_order = move.picking_id.sale_order_id
|
||||||
else:
|
else:
|
||||||
@@ -824,10 +893,12 @@ class ReStockMove(models.Model):
|
|||||||
continue
|
continue
|
||||||
product_name = ''
|
product_name = ''
|
||||||
logging.info('制造订单的产品 %s', production_id.product_id.name)
|
logging.info('制造订单的产品 %s', production_id.product_id.name)
|
||||||
match = re.search(r'(S\d{5}-\d)', production_id.product_id.name)
|
match = re.search(r'(S\d{5}-\d+)', production_id.product_id.name)
|
||||||
# 如果匹配成功,提取结果
|
# 如果匹配成功,提取结果
|
||||||
if match:
|
if match:
|
||||||
product_name = match.group(0)
|
product_name = match.group(0)
|
||||||
|
else:
|
||||||
|
product_name = production_id.product_id.name
|
||||||
if move.picking_id.sale_order_id:
|
if move.picking_id.sale_order_id:
|
||||||
sale_order = move.picking_id.sale_order_id
|
sale_order = move.picking_id.sale_order_id
|
||||||
else:
|
else:
|
||||||
@@ -848,7 +919,7 @@ class ReStockMove(models.Model):
|
|||||||
traceback_error = traceback.format_exc()
|
traceback_error = traceback.format_exc()
|
||||||
logging.error("零件图号 零件名称获取失败:%s" % traceback_error)
|
logging.error("零件图号 零件名称获取失败:%s" % traceback_error)
|
||||||
|
|
||||||
def _get_stock_move_values_Res(self, item, picking_type_id, group_id, move_dest_ids=False):
|
def _get_stock_move_values_Res(self, item, picking_type_id, group_id, move_dest_ids=False, product_uom_qty=1.0):
|
||||||
route_id = self.env.ref('sf_manufacturing.route_surface_technology_outsourcing').id
|
route_id = self.env.ref('sf_manufacturing.route_surface_technology_outsourcing').id
|
||||||
stock_rule = self.env['stock.rule'].sudo().search(
|
stock_rule = self.env['stock.rule'].sudo().search(
|
||||||
[('route_id', '=', route_id), ('picking_type_id', '=', picking_type_id)])
|
[('route_id', '=', route_id), ('picking_type_id', '=', picking_type_id)])
|
||||||
@@ -857,7 +928,7 @@ class ReStockMove(models.Model):
|
|||||||
'company_id': item.company_id.id,
|
'company_id': item.company_id.id,
|
||||||
'product_id': item.bom_id.bom_line_ids.product_id.id,
|
'product_id': item.bom_id.bom_line_ids.product_id.id,
|
||||||
'product_uom': item.bom_id.bom_line_ids.product_uom_id.id,
|
'product_uom': item.bom_id.bom_line_ids.product_uom_id.id,
|
||||||
'product_uom_qty': 1.0,
|
'product_uom_qty': product_uom_qty,
|
||||||
'location_id': stock_rule.location_src_id.id,
|
'location_id': stock_rule.location_src_id.id,
|
||||||
'location_dest_id': stock_rule.location_dest_id.id,
|
'location_dest_id': stock_rule.location_dest_id.id,
|
||||||
'origin': item.name,
|
'origin': item.name,
|
||||||
@@ -902,7 +973,7 @@ class ReStockMove(models.Model):
|
|||||||
'location_id': self.picking_id.location_id.id,
|
'location_id': self.picking_id.location_id.id,
|
||||||
'location_dest_id': self.picking_id.location_dest_id.id,
|
'location_dest_id': self.picking_id.location_dest_id.id,
|
||||||
'picking_id': self.picking_id.id,
|
'picking_id': self.picking_id.id,
|
||||||
'reserved_uom_qty': 1.0,
|
'reserved_uom_qty': self.product_uom_qty,
|
||||||
'lot_id': production_id.move_line_raw_ids.lot_id.id,
|
'lot_id': production_id.move_line_raw_ids.lot_id.id,
|
||||||
'company_id': self.env.company.id,
|
'company_id': self.env.company.id,
|
||||||
# 'workorder_id': '' if not sorted_workorders else sorted_workorders.id,
|
# 'workorder_id': '' if not sorted_workorders else sorted_workorders.id,
|
||||||
|
|||||||
11
sf_manufacturing/models/stock_warehouse_orderpoint.py
Normal file
11
sf_manufacturing/models/stock_warehouse_orderpoint.py
Normal file
@@ -0,0 +1,11 @@
|
|||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
# Part of SmartGo. See LICENSE file for full copyright and licensing details.
|
||||||
|
import base64
|
||||||
|
from io import BytesIO
|
||||||
|
from odoo import api, fields, models, SUPERUSER_ID, _
|
||||||
|
|
||||||
|
|
||||||
|
class StockWarehouseOrderpoint(models.Model):
|
||||||
|
_inherit = 'stock.warehouse.orderpoint'
|
||||||
|
origin = fields.Char(string='来源')
|
||||||
|
_order = 'create_date DESC'
|
||||||
@@ -110,12 +110,12 @@
|
|||||||
</xpath>
|
</xpath>
|
||||||
<xpath expr="//sheet//group//group[2]//label" position="before">
|
<xpath expr="//sheet//group//group[2]//label" position="before">
|
||||||
<!-- <field name="process_state"/> -->
|
<!-- <field name="process_state"/> -->
|
||||||
|
<field name="production_type" readonly="1"/>
|
||||||
<field name="state" readonly="1"/>
|
<field name="state" readonly="1"/>
|
||||||
<!-- <field name="process_state"/> -->
|
<!-- <field name="process_state"/> -->
|
||||||
|
|
||||||
</xpath>
|
</xpath>
|
||||||
<xpath expr="//sheet//group//group//div[3]" position="after">
|
<xpath expr="//sheet//group//group//div[3]" position="after">
|
||||||
<field name="production_type" readonly="1"/>
|
|
||||||
<field name="production_product_type" invisible="1"/>
|
<field name="production_product_type" invisible="1"/>
|
||||||
<field name="manual_quotation" readonly="1"
|
<field name="manual_quotation" readonly="1"
|
||||||
attrs="{'invisible': ['|', ('production_type', 'not in', ['自动化产线加工', '人工线下加工']), ('production_product_type', '!=', '成品')]}"/>
|
attrs="{'invisible': ['|', ('production_type', 'not in', ['自动化产线加工', '人工线下加工']), ('production_product_type', '!=', '成品')]}"/>
|
||||||
@@ -383,7 +383,7 @@
|
|||||||
<field name="process_parameters_id"
|
<field name="process_parameters_id"
|
||||||
attrs="{'readonly': [('id', '!=', False),('routing_tag', '=', 'standard')]}"
|
attrs="{'readonly': [('id', '!=', False),('routing_tag', '=', 'standard')]}"
|
||||||
string="参数" context="{'route_id':route_id,'production_id': production_id}"
|
string="参数" context="{'route_id':route_id,'production_id': production_id}"
|
||||||
options="{'no_create': True}" domain="[('routing_id', '=', 'route_id')]"/>
|
options="{'no_create': True}"/>
|
||||||
<field name="panel" readonly="1"/>
|
<field name="panel" readonly="1"/>
|
||||||
<field name="routing_tag" readonly="1" widget="badge"
|
<field name="routing_tag" readonly="1" widget="badge"
|
||||||
decoration-success="routing_tag == 'standard'"
|
decoration-success="routing_tag == 'standard'"
|
||||||
@@ -427,6 +427,7 @@
|
|||||||
<field name="programming_method"/>
|
<field name="programming_method"/>
|
||||||
<field name="current_programming_count"/>
|
<field name="current_programming_count"/>
|
||||||
<field name="target_production_id"/>
|
<field name="target_production_id"/>
|
||||||
|
<field name="apply_uid"/>
|
||||||
<field name="apply_time"/>
|
<field name="apply_time"/>
|
||||||
<field name="send_time"/>
|
<field name="send_time"/>
|
||||||
</tree>
|
</tree>
|
||||||
@@ -455,7 +456,7 @@
|
|||||||
<field name="inherit_id" ref="mrp.mrp_production_workorder_tree_editable_view"/>
|
<field name="inherit_id" ref="mrp.mrp_production_workorder_tree_editable_view"/>
|
||||||
<field name="arch" type="xml">
|
<field name="arch" type="xml">
|
||||||
<xpath expr="//tree" position="attributes">
|
<xpath expr="//tree" position="attributes">
|
||||||
<attribute name="default_order">create_date desc</attribute>
|
<attribute name="default_order">sequence,create_date desc</attribute>
|
||||||
<attribute name="decoration-warning">delivery_warning == 'warning'</attribute>
|
<attribute name="decoration-warning">delivery_warning == 'warning'</attribute>
|
||||||
<attribute name="decoration-danger">delivery_warning == 'overdue'</attribute>
|
<attribute name="decoration-danger">delivery_warning == 'overdue'</attribute>
|
||||||
</xpath>
|
</xpath>
|
||||||
@@ -602,6 +603,7 @@
|
|||||||
<field name="part_number"/>
|
<field name="part_number"/>
|
||||||
<field name="sale_order_id"/>
|
<field name="sale_order_id"/>
|
||||||
<field name="deadline_of_delivery" icon="fa-calendar" enable_counters="1" filter_domain="[('deadline_of_delivery', 'ilike', self)]"/>
|
<field name="deadline_of_delivery" icon="fa-calendar" enable_counters="1" filter_domain="[('deadline_of_delivery', 'ilike', self)]"/>
|
||||||
|
<field name="model_id"/>
|
||||||
</xpath>
|
</xpath>
|
||||||
<xpath expr="//field[@name='product_variant_attributes']" position="attributes">
|
<xpath expr="//field[@name='product_variant_attributes']" position="attributes">
|
||||||
<attribute name="invisible">1</attribute>
|
<attribute name="invisible">1</attribute>
|
||||||
|
|||||||
@@ -22,7 +22,7 @@
|
|||||||
<field name="is_repeat"/>
|
<field name="is_repeat"/>
|
||||||
<field name="reserved_duration"/>
|
<field name="reserved_duration"/>
|
||||||
</field>
|
</field>
|
||||||
<xpath expr="//notebook/page[1]" position="before">
|
<!-- <xpath expr="//notebook/page[1]" position="before">
|
||||||
<page string="可选工艺参数">
|
<page string="可选工艺参数">
|
||||||
<field name="optional_process_parameters">
|
<field name="optional_process_parameters">
|
||||||
<tree editable="bottom">
|
<tree editable="bottom">
|
||||||
@@ -30,9 +30,9 @@
|
|||||||
<field name="is_delete_button" invisible="1"/>
|
<field name="is_delete_button" invisible="1"/>
|
||||||
<field name="code" attrs="{'readonly': True}"/>
|
<field name="code" attrs="{'readonly': True}"/>
|
||||||
<field name="name" required="1"/>
|
<field name="name" required="1"/>
|
||||||
<field name="service_products" domain="[('detailed_type', '=', 'service'),('server_product_process_parameters_id', '=', False)]"/>
|
<field name="service_products" domain="[('detailed_type', '=', 'service'),('server_product_process_parameters_id', '=', False)]"/> -->
|
||||||
<!-- 按钮列 -->
|
<!-- 按钮列 -->
|
||||||
<button name="action_create_service_product" string="创建服务产品" type="object"
|
<!-- <button name="action_create_service_product" string="创建服务产品" type="object"
|
||||||
class="btn-primary"
|
class="btn-primary"
|
||||||
attrs="{'invisible': [('is_product_button', '=', True)]}" context="{'default_process_parameter_id':id}"/>
|
attrs="{'invisible': [('is_product_button', '=', True)]}" context="{'default_process_parameter_id':id}"/>
|
||||||
<button name="action_hide_service_products" string="删除" type="object"
|
<button name="action_hide_service_products" string="删除" type="object"
|
||||||
@@ -41,7 +41,7 @@
|
|||||||
</tree>
|
</tree>
|
||||||
</field>
|
</field>
|
||||||
</page>
|
</page>
|
||||||
</xpath>
|
</xpath> -->
|
||||||
</field>
|
</field>
|
||||||
</record>
|
</record>
|
||||||
</data>
|
</data>
|
||||||
|
|||||||
@@ -4,30 +4,32 @@
|
|||||||
name="Manufacturing"
|
name="Manufacturing"
|
||||||
groups="mrp.group_mrp_user,mrp.group_mrp_manager,sf_base.group_sf_mrp_user,sf_base.group_sf_mrp_manager"
|
groups="mrp.group_mrp_user,mrp.group_mrp_manager,sf_base.group_sf_mrp_user,sf_base.group_sf_mrp_manager"
|
||||||
web_icon="mrp,static/description/icon.svg"
|
web_icon="mrp,static/description/icon.svg"
|
||||||
sequence="145">
|
sequence="145"/>
|
||||||
|
|
||||||
<menuitem id="mrp.menu_mrp_manufacturing"
|
<menuitem id="mrp.menu_mrp_manufacturing"
|
||||||
name="Operations"
|
name="Operations"
|
||||||
sequence="10"/>
|
parent="mrp.menu_mrp_root"
|
||||||
|
sequence="10"/>
|
||||||
|
|
||||||
<menuitem id="mrp.mrp_planning_menu_root"
|
<menuitem id="mrp.mrp_planning_menu_root"
|
||||||
name="Planning"
|
name="Planning"
|
||||||
sequence="15"/>
|
parent="mrp.menu_mrp_root"
|
||||||
|
sequence="15"/>
|
||||||
|
|
||||||
<menuitem id="mrp.enu_mrp_bom"
|
<menuitem id="mrp.menu_mrp_bom"
|
||||||
name="Products"
|
name="Products"
|
||||||
sequence="20"/>
|
parent="mrp.menu_mrp_root"
|
||||||
|
sequence="20"/>
|
||||||
|
|
||||||
<menuitem id="mrp.menu_mrp_reporting"
|
<menuitem id="mrp.menu_mrp_reporting"
|
||||||
name="Reporting"
|
name="Reporting"
|
||||||
sequence="25"/>
|
parent="mrp.menu_mrp_root"
|
||||||
|
sequence="25"/>
|
||||||
<menuitem id="mrp.menu_mrp_configuration"
|
|
||||||
name="Configuration"
|
|
||||||
groups="mrp.group_mrp_manager,sf_base.group_sf_mrp_manager"
|
|
||||||
sequence="100"/>
|
|
||||||
|
|
||||||
</menuitem>
|
|
||||||
|
|
||||||
|
<menuitem id="mrp.menu_mrp_configuration"
|
||||||
|
name="Configuration"
|
||||||
|
parent="mrp.menu_mrp_root"
|
||||||
|
groups="mrp.group_mrp_manager,sf_base.group_sf_mrp_manager"
|
||||||
|
sequence="100"/>
|
||||||
|
|
||||||
</odoo>
|
</odoo>
|
||||||
|
|||||||
@@ -1,15 +1,15 @@
|
|||||||
<?xml version="1.0" encoding="utf-8"?>
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
<odoo>
|
<odoo>
|
||||||
<!-- 工作中心看板 -->
|
<!-- 工作中心看板 -->
|
||||||
<record id="mrp_production_view_form_inherit_maintenance" model="ir.ui.view">
|
<!-- <record id="mrp_production_view_form_inherit_maintenance" model="ir.ui.view">
|
||||||
<field name="name">mrp.production.view.form.inherit.maintenance</field>
|
<field name="name">mrp.production.view.form.inherit.maintenance</field>
|
||||||
<field name="model">mrp.production</field>
|
<field name="model">mrp.production</field>
|
||||||
<field name="inherit_id" ref="mrp.mrp_production_form_view"/>
|
<field name="inherit_id" ref="mrp.mrp_production_form_view"/>
|
||||||
<field name="arch" type="xml">
|
<field name="arch" type="xml"> -->
|
||||||
<!-- <button name="action_cancel" position="before"> -->
|
<!-- <button name="action_cancel" position="before"> -->
|
||||||
<!-- <button name="button_maintenance_req" type="object" string="维修请求"/> -->
|
<!-- <button name="button_maintenance_req" type="object" string="维修请求"/> -->
|
||||||
<!-- </button> -->
|
<!-- </button> -->
|
||||||
<div name="button_box" position="inside">
|
<!-- <div name="button_box" position="inside">
|
||||||
<button name="open_maintenance_request_mo" type="object" class="oe_stat_button" icon="fa-wrench"
|
<button name="open_maintenance_request_mo" type="object" class="oe_stat_button" icon="fa-wrench"
|
||||||
attrs="{'invisible': [('maintenance_count', '=', 0)]}"
|
attrs="{'invisible': [('maintenance_count', '=', 0)]}"
|
||||||
context="{'search_default_production_id': active_id}">
|
context="{'search_default_production_id': active_id}">
|
||||||
@@ -22,7 +22,7 @@
|
|||||||
</button>
|
</button>
|
||||||
</div>
|
</div>
|
||||||
</field>
|
</field>
|
||||||
</record>
|
</record> -->
|
||||||
|
|
||||||
<record id="custom_model_form_view_inherit" model="ir.ui.view">
|
<record id="custom_model_form_view_inherit" model="ir.ui.view">
|
||||||
<field name="name">custom.model.form.view.inherit</field>
|
<field name="name">custom.model.form.view.inherit</field>
|
||||||
@@ -451,6 +451,7 @@
|
|||||||
</div>
|
</div>
|
||||||
<field name="product_id" position="after">
|
<field name="product_id" position="after">
|
||||||
<field name="model_file" string="产品模型" readonly="1" widget="Viewer3D" attrs="{'invisible': [('model_file', '=', False)]}"/>
|
<field name="model_file" string="产品模型" readonly="1" widget="Viewer3D" attrs="{'invisible': [('model_file', '=', False)]}"/>
|
||||||
|
<field name="model_id" readonly="1"/>
|
||||||
<field name="glb_url" widget="Viewer3D" string="模型" readonly="1" force_save="1"
|
<field name="glb_url" widget="Viewer3D" string="模型" readonly="1" force_save="1"
|
||||||
attrs="{'invisible': [('glb_url', '=', False)]}"/>
|
attrs="{'invisible': [('glb_url', '=', False)]}"/>
|
||||||
</field>
|
</field>
|
||||||
|
|||||||
@@ -144,7 +144,7 @@
|
|||||||
statusbar_visible="pending,waiting,ready,progress,to be detected,done,rework"/>
|
statusbar_visible="pending,waiting,ready,progress,to be detected,done,rework"/>
|
||||||
</xpath>
|
</xpath>
|
||||||
<xpath expr="//div[@name='button_box']" position="inside">
|
<xpath expr="//div[@name='button_box']" position="inside">
|
||||||
<button type="object" name="action_view_pr_mrp_workorder" class="oe_stat_button"
|
<!-- <button type="object" name="action_view_pr_mrp_workorder" class="oe_stat_button"
|
||||||
icon="fa-credit-card"
|
icon="fa-credit-card"
|
||||||
groups="base.group_user,sf_base.group_sf_order_user"
|
groups="base.group_user,sf_base.group_sf_order_user"
|
||||||
attrs="{'invisible': [('pr_mp_count', '=', 0)]}">
|
attrs="{'invisible': [('pr_mp_count', '=', 0)]}">
|
||||||
@@ -154,7 +154,7 @@
|
|||||||
</span>
|
</span>
|
||||||
<span class="o_stat_text">采购申请</span>
|
<span class="o_stat_text">采购申请</span>
|
||||||
</div>
|
</div>
|
||||||
</button>
|
</button> -->
|
||||||
<button type="object" name="action_view_surface_technics_purchase" class="oe_stat_button"
|
<button type="object" name="action_view_surface_technics_purchase" class="oe_stat_button"
|
||||||
icon="fa-credit-card"
|
icon="fa-credit-card"
|
||||||
groups="base.group_user,sf_base.group_sf_order_user"
|
groups="base.group_user,sf_base.group_sf_order_user"
|
||||||
@@ -677,8 +677,9 @@
|
|||||||
<field name="inherit_id" ref="mrp.view_mrp_production_work_order_search"/>
|
<field name="inherit_id" ref="mrp.view_mrp_production_work_order_search"/>
|
||||||
<field name="arch" type="xml">
|
<field name="arch" type="xml">
|
||||||
<field name="product_id" position="after">
|
<field name="product_id" position="after">
|
||||||
<field name="part_number" string="成品零件图号"/>
|
<field name="part_number" string="零件图号"/>
|
||||||
<field name="model_id" string="模型id"/>
|
<field name="part_name" string="零件名称"/>
|
||||||
|
<field name="model_id" string="模型ID"/>
|
||||||
</field>
|
</field>
|
||||||
<xpath expr="//filter[@name='progress']" position="after">
|
<xpath expr="//filter[@name='progress']" position="after">
|
||||||
<filter string="待检测" name="state" domain="[('state','=','to be detected')]"/>
|
<filter string="待检测" name="state" domain="[('state','=','to be detected')]"/>
|
||||||
|
|||||||
@@ -4,10 +4,24 @@
|
|||||||
<record model="ir.ui.view" id="view_product_template_form_inherit_sf_manufacturing">
|
<record model="ir.ui.view" id="view_product_template_form_inherit_sf_manufacturing">
|
||||||
<field name="name">product.template.product.form.inherit.sf_manufacture</field>
|
<field name="name">product.template.product.form.inherit.sf_manufacture</field>
|
||||||
<field name="model">product.template</field>
|
<field name="model">product.template</field>
|
||||||
<field name="inherit_id" ref="sf_dlm_management.view_product_template_only_form_inherit_sf"/>
|
<field name="inherit_id" ref="sf_sale.view_product_template_form_inherit_sf"/>
|
||||||
<field name="arch" type="xml">
|
<field name="arch" type="xml">
|
||||||
<xpath expr="//page[@name='general_information']/group/group[@name='group_standard_price']/field[@name='barcode']" position="after">
|
<xpath expr="//page[@name='general_information']/group/group[@name='group_standard_price']/field[@name='product_tag_ids']" position="after">
|
||||||
<field name="model_id" readonly="1"/>
|
<field name="categ_type" invisible="1"/>
|
||||||
|
<field name="model_id" readonly="1" attrs="{'invisible': [('categ_type', '!=', '成品')]}"/>
|
||||||
|
</xpath>
|
||||||
|
</field>
|
||||||
|
</record>
|
||||||
|
|
||||||
|
<record id="product_template_search_inherit_sf_manufacturing" model="ir.ui.view">
|
||||||
|
<field name="name">product.template.search</field>
|
||||||
|
<field name="model">product.template</field>
|
||||||
|
<field name="inherit_id" ref="product.product_template_search_view"/>
|
||||||
|
<field name="arch" type="xml">
|
||||||
|
<xpath expr="//field[@name='categ_id']" position="after">
|
||||||
|
<field name="part_number" string="零件图号"/>
|
||||||
|
<field name="part_name" string="零件名称"/>
|
||||||
|
<field name="model_id" string="模型ID"/>
|
||||||
</xpath>
|
</xpath>
|
||||||
</field>
|
</field>
|
||||||
</record>
|
</record>
|
||||||
|
|||||||
@@ -73,6 +73,7 @@
|
|||||||
<xpath expr="//field[@name='picking_type_id']" position="after">
|
<xpath expr="//field[@name='picking_type_id']" position="after">
|
||||||
<field name="part_numbers" string="零件图号" filter_domain="[('part_numbers', 'ilike', self)]"/>
|
<field name="part_numbers" string="零件图号" filter_domain="[('part_numbers', 'ilike', self)]"/>
|
||||||
<field name="part_names" string="零件名称" filter_domain="[('part_names', 'ilike', self)]"/>
|
<field name="part_names" string="零件名称" filter_domain="[('part_names', 'ilike', self)]"/>
|
||||||
|
<field name="model_id" string="模型ID" filter_domain="[('model_id', 'ilike', self)]"/>
|
||||||
</xpath>
|
</xpath>
|
||||||
</field>
|
</field>
|
||||||
</record>
|
</record>
|
||||||
|
|||||||
16
sf_manufacturing/views/stock_warehouse_orderpoint.xml
Normal file
16
sf_manufacturing/views/stock_warehouse_orderpoint.xml
Normal file
@@ -0,0 +1,16 @@
|
|||||||
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
|
<odoo>
|
||||||
|
<data>
|
||||||
|
<record id="view_warehouse_orderpoint_tree_editable_inherit" model="ir.ui.view">
|
||||||
|
<field name="name">补货</field>
|
||||||
|
<field name="model">stock.warehouse.orderpoint</field>
|
||||||
|
<field name="inherit_id" ref="stock.view_warehouse_orderpoint_tree_editable"/>
|
||||||
|
<field name="arch" type="xml">
|
||||||
|
<xpath expr="//field[@name='qty_to_order']" position="after">
|
||||||
|
<field name="origin"/>
|
||||||
|
</xpath>
|
||||||
|
</field>
|
||||||
|
</record>
|
||||||
|
<!-- 继承补货单的搜索视图 -->
|
||||||
|
</data>
|
||||||
|
</odoo>
|
||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user