Compare commits
1000 Commits
release/re
...
feature/前端
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
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 | ||
|
|
83feb78f43 | ||
|
|
ac09794b10 | ||
|
|
3f88b11a18 | ||
|
|
b97acfb181 | ||
|
|
6767b693f5 | ||
|
|
7b77d846c3 | ||
|
|
5e3bce1931 | ||
|
|
ef5c2649c9 | ||
|
|
5400c9ec69 | ||
|
|
7c7876e96f | ||
|
|
5438ff6df1 | ||
|
|
05b5e9cfd9 | ||
|
|
86dfd437cc | ||
|
|
3a92e4cfd8 | ||
|
|
bff0ff9401 | ||
|
|
9b3fe5c070 | ||
|
|
92d593bf24 | ||
|
|
dc6fb4ddda | ||
|
|
a84f8636a4 | ||
|
|
7c26045377 | ||
|
|
da579a15b3 | ||
|
|
c9ee0af25a | ||
|
|
151bc5da4f | ||
|
|
1015af483c | ||
|
|
bb3971c93e | ||
|
|
aa9efe932e | ||
|
|
8e09e9715c | ||
|
|
30a55658de | ||
|
|
7ecdc31db4 | ||
|
|
4234a6bbd9 | ||
|
|
8fcc436ce5 | ||
|
|
53a676bc93 | ||
|
|
c6f625fe44 | ||
|
|
397c35867b | ||
|
|
f843610872 | ||
|
|
41e4d2656a | ||
|
|
4272d2855f | ||
|
|
e63a12cbc1 | ||
|
|
2867264e88 | ||
|
|
7c2ddd8a0c | ||
|
|
4daddcff61 | ||
|
|
666c0167d7 | ||
|
|
6b7f1aedbe | ||
|
|
4234494f08 | ||
|
|
cbc73ee8e3 | ||
|
|
ec3d2a9239 | ||
|
|
a55da93dc4 | ||
|
|
73dce5e75e | ||
|
|
9b009ec5a3 | ||
|
|
7d70184a5d | ||
|
|
9f9c08b1f0 | ||
|
|
5886b4e132 | ||
|
|
77478fd173 | ||
|
|
2154533ad5 | ||
|
|
eab7bf04c3 | ||
|
|
ba910f011e | ||
|
|
c43d8ee280 | ||
|
|
acc04f396d | ||
|
|
3425376705 | ||
|
|
7849ce6342 | ||
|
|
69e8bf67d1 | ||
|
|
722bdc251d | ||
|
|
0403a2d223 | ||
|
|
7a4c3f155c | ||
|
|
acd8d9c758 | ||
|
|
d33a801f89 | ||
|
|
a74c07f0bb | ||
|
|
9f1cac0789 | ||
|
|
fcd0319a9c | ||
|
|
ed2c76adf1 | ||
|
|
1d45486e20 | ||
|
|
d209d40356 | ||
|
|
3355731632 | ||
|
|
ddd902cad8 | ||
|
|
e5566edd7c | ||
|
|
ea0f9d6ee2 | ||
|
|
79006e062a | ||
|
|
ee9783d4a6 | ||
|
|
ff7c4a9ce7 | ||
|
|
48b94e8d84 | ||
|
|
a395994a1c | ||
|
|
43c29e30b0 | ||
|
|
be639d7631 | ||
|
|
66c745af3c | ||
|
|
29ee275840 | ||
|
|
f92b844c16 | ||
|
|
63870ed7a9 | ||
|
|
558c99f48b | ||
|
|
1264a4305e | ||
|
|
a4f8183654 | ||
|
|
3ea8e9d35f | ||
|
|
3fb2890bf1 | ||
|
|
0c3ba5c172 | ||
|
|
1b22d02700 | ||
|
|
962931bc9d | ||
|
|
9cfc8418ec | ||
|
|
e4af2aacb8 | ||
|
|
c67944f689 | ||
|
|
9c713c2a37 | ||
|
|
53d4be596f | ||
|
|
195302b67f | ||
|
|
6cb5e909d4 | ||
|
|
cb98f5cf25 | ||
|
|
f87a6690fe | ||
|
|
b041feb297 | ||
|
|
06e9d5a538 | ||
|
|
b9aac6c558 | ||
|
|
2de0e9f02f | ||
|
|
722b601890 | ||
|
|
20722c12b8 | ||
|
|
2100ee9590 | ||
|
|
56f2ea0356 | ||
|
|
6ff8395916 | ||
|
|
bc85c457ad | ||
|
|
085e6359ce | ||
|
|
7eeea92a3e | ||
|
|
d672f3f4d7 | ||
|
|
c79cf2e5ad | ||
|
|
33a5fc0ff4 | ||
|
|
2dcaa25952 | ||
|
|
ba88070fad | ||
|
|
3f940992be | ||
|
|
197ae6bc01 | ||
|
|
07336326ce | ||
|
|
c93553e78e | ||
|
|
0db31f7936 | ||
|
|
61a3cf606e | ||
|
|
91d79008e1 | ||
|
|
8cdf77f609 | ||
|
|
37edc858c2 | ||
|
|
17fdf20e03 | ||
|
|
a896108638 | ||
|
|
026697f363 | ||
|
|
ec934abc42 | ||
|
|
cc030957fb | ||
|
|
87786dbd80 | ||
|
|
9d0ffd23b2 | ||
|
|
3fb56f15c8 | ||
|
|
bdf4696c08 | ||
|
|
6c926bf081 | ||
|
|
ddb0c304b9 | ||
|
|
cf8c14e738 | ||
|
|
6bd6816495 | ||
|
|
2bae98950e | ||
|
|
ec379a7541 | ||
|
|
119acf1543 | ||
|
|
0f6f1aae24 | ||
|
|
c40ecfb6ce | ||
|
|
a51a4c2fbb | ||
|
|
315e2aa03d | ||
|
|
10bea40159 | ||
|
|
78ba8d0ead | ||
|
|
e61742cc5b | ||
|
|
d0d1a640d9 | ||
|
|
e613a2f283 | ||
|
|
88e83c0e14 | ||
|
|
f912a81e7b | ||
|
|
e09226e966 | ||
|
|
8a7a90ff0d | ||
|
|
d05a3606c3 | ||
|
|
4b5c3b383b | ||
|
|
1c34f2f15c | ||
|
|
80f259651c | ||
|
|
1c57ee0be1 | ||
|
|
c732bbad62 | ||
|
|
5285fcd066 | ||
|
|
d318d8cb32 | ||
|
|
5b9dc05653 | ||
|
|
a513592b21 | ||
|
|
e686ea9469 | ||
|
|
fd5ff0904e | ||
|
|
651918c51c | ||
|
|
7b13dfcc0e | ||
|
|
60e139d5e0 | ||
|
|
8d5ea0ae19 | ||
|
|
d7597359ba | ||
|
|
f8309bfaba | ||
|
|
7ed756f922 | ||
|
|
e837b84a50 | ||
|
|
e7cb100ab1 | ||
|
|
d00c9dd38c | ||
|
|
1d857be16a | ||
|
|
5914e4ca6e | ||
|
|
d96970fb96 | ||
|
|
0af9064fce | ||
|
|
21148ae74b | ||
|
|
1a3590b6b6 | ||
|
|
b55c6c1fe7 | ||
|
|
8c61dcac29 | ||
|
|
41d4e9785f | ||
|
|
5bf86930e9 | ||
|
|
7ad9885377 | ||
|
|
71433c18b7 | ||
|
|
4b60ad307b | ||
|
|
1a5c8e5f56 | ||
|
|
8b1e12eb9f | ||
|
|
dbf2257a88 | ||
|
|
f34c01d1b0 | ||
|
|
9c73062593 | ||
|
|
878fef18df | ||
|
|
8348c4fc48 | ||
|
|
8643fb2385 | ||
|
|
77744e75d7 | ||
|
|
4cbcf08da8 | ||
|
|
af2a589679 | ||
|
|
4b6f04aa9d | ||
|
|
4634d43012 | ||
|
|
dd6e8b6707 | ||
|
|
55337815c7 | ||
|
|
0ccb7cb3d1 | ||
|
|
d2155b17b4 | ||
|
|
11a5217430 | ||
|
|
855e0eb1c2 | ||
|
|
6a70f3b88a | ||
|
|
236158d556 | ||
|
|
1a67c5d1e3 | ||
|
|
bdb8763fea | ||
|
|
78ed512699 | ||
|
|
2e74c76e07 | ||
|
|
40dcd11da8 | ||
|
|
f9b40be428 | ||
|
|
4ca655ad51 | ||
|
|
a2ed102895 | ||
|
|
8ec746858c | ||
|
|
03539c7ed3 | ||
|
|
4d38a11c3c | ||
|
|
6312bb988a | ||
|
|
2b082a0dd7 | ||
|
|
3e9a11dd7b | ||
|
|
f011c1efda | ||
|
|
fe9548a0d1 | ||
|
|
ac2d81285e | ||
|
|
73ae2cd5c3 | ||
|
|
0347eb48e4 | ||
|
|
0aae15cbce | ||
|
|
d888be06c4 | ||
|
|
2275f7a384 | ||
|
|
5a7d70fb6b | ||
|
|
1256f5ecb5 | ||
|
|
4dfac9e96f | ||
|
|
8ea8bf1f48 | ||
|
|
29f143a547 | ||
|
|
2eaf8aef8f | ||
|
|
df78019226 | ||
|
|
6ad945b720 | ||
|
|
3285d4da57 | ||
|
|
8d1466485a | ||
|
|
b39281d057 | ||
|
|
60539462a0 | ||
|
|
fd9018a4c8 | ||
|
|
81b425ae0c | ||
|
|
f8e8615dc8 | ||
|
|
2b2da79e33 | ||
|
|
01ac8a19b6 | ||
|
|
20a8ca6146 | ||
|
|
62cd60ec06 | ||
|
|
33fabc068a | ||
|
|
db4dd33709 | ||
|
|
4acb0fa0ba | ||
|
|
c9ba3a07bf | ||
|
|
89a2260adc | ||
|
|
c5ad94c5f3 | ||
|
|
02e0a792d4 | ||
|
|
e6d54a3eba | ||
|
|
43c6686240 | ||
|
|
9b73ae26e8 | ||
|
|
a0606842e5 | ||
|
|
d014ba980b | ||
|
|
5c35eae859 | ||
|
|
9df76253de | ||
|
|
8ba71ea8af | ||
|
|
fde28bed8a | ||
|
|
8fed12f7bb | ||
|
|
489d7030f4 | ||
|
|
16ae845ad9 | ||
|
|
2c1d90ae63 | ||
|
|
8e21434fba | ||
|
|
956cefbd9f | ||
|
|
85789d137b | ||
|
|
6e731cfadc | ||
|
|
f51f8bebb2 | ||
|
|
e10648ad07 | ||
|
|
2bd18def8f | ||
|
|
61b2b05367 | ||
|
|
d2b02bb6f7 | ||
|
|
e7b4f51736 | ||
|
|
8832b01408 | ||
|
|
a5c5b499f1 | ||
|
|
0e1c44c3ac | ||
|
|
4212ed763b | ||
|
|
a6ac46617d | ||
|
|
43ba241b42 | ||
|
|
5ffbe4c6fc | ||
|
|
5ae167c133 | ||
|
|
0702e1dd51 | ||
|
|
d9d434d994 | ||
|
|
08fa26d9c8 | ||
|
|
e861897527 | ||
|
|
4ba3a56794 | ||
|
|
4c995d00d0 | ||
|
|
30b01b6e2c | ||
|
|
19a66d2c81 | ||
|
|
b2b9120577 | ||
|
|
ad8f63570d | ||
|
|
3bb909c2ee | ||
|
|
e8b22cedea | ||
|
|
d202e7ee93 | ||
|
|
da19b86bf5 | ||
|
|
2876d56803 | ||
|
|
e1fd4c465e | ||
|
|
dc318769af | ||
|
|
31820dfef1 | ||
|
|
1cb46bb1e1 | ||
|
|
80e9085e29 | ||
|
|
311b95bca5 | ||
|
|
d1f8bdb1f4 | ||
|
|
08267ba1e6 | ||
|
|
c6f1bc80a7 | ||
|
|
384761514b | ||
|
|
80118b61c2 | ||
|
|
eab3ba9478 | ||
|
|
c2be45f204 | ||
|
|
86d2fb1ac2 | ||
|
|
f28d07e58f | ||
|
|
7124eebabe | ||
|
|
94d0c14e1f | ||
|
|
9f3791bd6b | ||
|
|
ae1028972c | ||
|
|
3f6f9bb709 | ||
|
|
b86d15c5cb | ||
|
|
52e68ffa4c | ||
|
|
f85f614190 | ||
|
|
2da22d5f19 | ||
|
|
98644a4b57 | ||
|
|
6c879e4af3 | ||
|
|
cf060d0d6c | ||
|
|
49a1ec353a | ||
|
|
134c68abc8 | ||
|
|
d6a7a3c919 | ||
|
|
b89cfb899b | ||
|
|
db661e76f5 | ||
|
|
bf8ff7199e | ||
|
|
0ff0cc5fbd | ||
|
|
813939a6c2 | ||
|
|
221e25f581 | ||
|
|
e4ed431167 | ||
|
|
1a511b93cf | ||
|
|
9093770ce1 | ||
|
|
37173968fd | ||
|
|
a75e236f1f | ||
|
|
e1177a44e8 | ||
|
|
470482b7e2 | ||
|
|
27702fe46e | ||
|
|
8c793c6ad9 | ||
|
|
0b067f9999 | ||
|
|
5b51cc3de4 | ||
|
|
625188d46d | ||
|
|
6178ad0f8e | ||
|
|
9e939467e5 | ||
|
|
6c89a2c4fb | ||
|
|
dc5e70c118 | ||
|
|
b05492615f | ||
|
|
03d6f9a32e | ||
|
|
8634c1fc40 | ||
|
|
32ed0e9693 | ||
|
|
e1c535f907 | ||
|
|
a556c21196 | ||
|
|
623ebe3ec3 | ||
|
|
b1821d3ed6 | ||
|
|
a8d8bcbcee | ||
|
|
da06688571 | ||
|
|
e7d8587a32 | ||
|
|
b53ea2e0e9 | ||
|
|
5b979ffc34 | ||
|
|
c01451336d | ||
|
|
e5fc3dfe62 | ||
|
|
08cd1a176b | ||
|
|
2db5068e85 | ||
|
|
bab21d7b76 | ||
|
|
6bdaad8718 | ||
|
|
94441422cd | ||
|
|
5f48dad86c | ||
|
|
c955953335 | ||
|
|
f5bf727b34 | ||
|
|
c55231555c | ||
|
|
9220c4b7c4 | ||
|
|
e4606ece75 | ||
|
|
972c06cdc1 | ||
|
|
70b21c607e | ||
|
|
f8bfb7ba40 | ||
|
|
b2c3694a23 | ||
|
|
be0df6e66b | ||
|
|
242f9c9050 | ||
|
|
4fe7300ec0 | ||
|
|
307bfa19f2 | ||
|
|
83b051e0b9 | ||
|
|
5bbcd8ec23 | ||
|
|
282458c945 | ||
|
|
b211ffdff9 | ||
|
|
cb445393ce | ||
|
|
25ba61607f | ||
|
|
f6e87493d3 | ||
|
|
8c4c65a00f | ||
|
|
3642398724 | ||
|
|
23bcf13d95 | ||
|
|
6b78cb72b3 | ||
|
|
fde508c563 | ||
|
|
b65682f5c9 | ||
|
|
bfe9c51d57 | ||
|
|
e0015f9d6b | ||
|
|
24166c2e71 | ||
|
|
8fb90c1c35 | ||
|
|
e141f0af2c | ||
|
|
f8b5871912 | ||
|
|
40a7c14b81 | ||
|
|
4618c83b2f | ||
|
|
dee616fbef | ||
|
|
a2868a2581 | ||
|
|
e89400f04e | ||
|
|
5e5c7e5512 | ||
|
|
117d90bd5c | ||
|
|
fb19a19b25 | ||
|
|
1311669814 | ||
|
|
8347a8b029 | ||
|
|
051f8128e9 | ||
|
|
2cdad0cec4 | ||
|
|
7df5bea5b1 | ||
|
|
40965af4c0 | ||
|
|
21876f4b9d | ||
|
|
5a8df0a1ae | ||
|
|
d280d2b776 | ||
|
|
323a204a48 | ||
|
|
3ce7f1771a | ||
|
|
e1dc2bde7e | ||
|
|
495f1962df | ||
|
|
221eb7c3a1 | ||
|
|
af529d21ec | ||
|
|
0a1b48ed93 | ||
|
|
db881ac594 | ||
|
|
61776ca6a3 | ||
|
|
6da31b4f48 | ||
|
|
84f74ae09f | ||
|
|
561b515242 | ||
|
|
71e79502f9 | ||
|
|
67f3c29af3 | ||
|
|
5d703b8b73 | ||
|
|
8f61f258b1 | ||
|
|
a864845d2b | ||
|
|
7755cc3982 | ||
|
|
34565c5d79 | ||
|
|
0f537a3158 | ||
|
|
7f99b4ba8b | ||
|
|
4138903bff | ||
|
|
4200fdbf3b | ||
|
|
a6a2e53111 | ||
|
|
214f45850e | ||
|
|
c23aa4de75 | ||
|
|
f66ff6339d | ||
|
|
f9525beb65 | ||
|
|
35bcfa6fa1 | ||
|
|
7b6dda3d75 | ||
|
|
954ff6b848 | ||
|
|
af3a2880e8 | ||
|
|
424a496046 | ||
|
|
bd2ba3bb49 | ||
|
|
2d4926f8b7 | ||
|
|
9d84b68525 | ||
|
|
3baf3e60e8 | ||
|
|
16dbfc3867 | ||
|
|
6de31608d1 | ||
|
|
73ce21ef99 | ||
|
|
c632926348 | ||
|
|
a421055348 | ||
|
|
cd114d183b | ||
|
|
20efa0bdab | ||
|
|
839b3c981d | ||
|
|
395db2cf5f | ||
|
|
1ca069b149 | ||
|
|
061714413c | ||
|
|
f780d47562 | ||
|
|
082e25feae | ||
|
|
ef5eea1845 | ||
|
|
359eae14cc | ||
|
|
528d63123f | ||
|
|
a11329eaf8 | ||
|
|
d571b77915 | ||
|
|
b94adc9704 | ||
|
|
2568e63712 | ||
|
|
7ddcfc6226 | ||
|
|
3543531f61 | ||
|
|
04b255d5c6 | ||
|
|
24897f07f8 | ||
|
|
53779b89a7 | ||
|
|
326dcc8b87 | ||
|
|
9d042dc61e | ||
|
|
a1a94867f0 | ||
|
|
7ba415968b | ||
|
|
e10483b87b | ||
|
|
83699fcae6 | ||
|
|
0a666f568d | ||
|
|
b2b8762138 | ||
|
|
7c48e6b186 | ||
|
|
ad8ec770b6 | ||
|
|
e87689da33 | ||
|
|
3cc3c48ab3 | ||
|
|
dff0841317 | ||
|
|
0d8445e444 | ||
|
|
f5da36a82c | ||
|
|
da489555b0 | ||
|
|
29337bfceb | ||
|
|
c6ae4d933c | ||
|
|
d577630657 | ||
|
|
5b084624df | ||
|
|
5e51ee8db3 | ||
|
|
de1bdbe18b | ||
|
|
78738ed8aa | ||
|
|
8237d04f32 | ||
|
|
ef0d05a29d | ||
|
|
e6e740a6c7 | ||
|
|
b8bec37e15 | ||
|
|
003a115084 | ||
|
|
9ca7f75ef7 | ||
|
|
7654a60f2f | ||
|
|
bd43de74dd | ||
|
|
2cc7386027 | ||
|
|
4a92e39b0d | ||
|
|
34456c3506 | ||
|
|
dedc820b50 | ||
|
|
dc550d1be5 | ||
|
|
1541326dfc | ||
|
|
0d7f348194 | ||
|
|
2ccedc95f2 | ||
|
|
307f1b221f | ||
|
|
674af1d11d | ||
|
|
b276f616e5 | ||
|
|
cb645aa1b9 | ||
|
|
e12755783c | ||
|
|
1215d62e76 | ||
|
|
00c9ad423c | ||
|
|
826d5b1d58 | ||
|
|
9a67caca74 | ||
|
|
9c5ecdfe76 | ||
|
|
0a65c29286 | ||
|
|
2409dab8b0 | ||
|
|
f2415ae80d | ||
|
|
7d4314abc7 | ||
|
|
28a3d52aea | ||
|
|
7e93586f69 | ||
|
|
e8fc38e6ed | ||
|
|
3d355ec303 | ||
|
|
52e585e637 | ||
|
|
05dac9fb0c | ||
|
|
d3a1e9b341 | ||
|
|
8db80e1ed6 | ||
|
|
fc378df597 | ||
|
|
0575d89227 | ||
|
|
39afc99b8f | ||
|
|
f230ad55fb |
@@ -118,6 +118,26 @@ patch(ListRenderer.prototype, 'jikimo_frontend.ListRenderer', {
|
|||||||
owl.onPatched(() => {
|
owl.onPatched(() => {
|
||||||
this.listherHeaderBodyNum()
|
this.listherHeaderBodyNum()
|
||||||
})
|
})
|
||||||
|
const treeModifiers = this.getFieldModifiers(this.props.archInfo.__rawArch);
|
||||||
|
|
||||||
|
if(treeModifiers) {
|
||||||
|
if(treeModifiers.merge_fields) {
|
||||||
|
this.props.merge_key = treeModifiers.merge_key;
|
||||||
|
this.props.merge_fields = treeModifiers.merge_fields.split(',');
|
||||||
|
const data = this.setColumns(this.props.merge_key);
|
||||||
|
owl.onMounted(() => {
|
||||||
|
this.mergeColumns(this.props.merge_fields, data)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
if(treeModifiers.pacthResize) {
|
||||||
|
|
||||||
|
owl.onPatched(() => {
|
||||||
|
this.columnWidths = null;
|
||||||
|
this.freezeColumnWidths();
|
||||||
|
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
return this._super(...arguments);
|
return this._super(...arguments);
|
||||||
},
|
},
|
||||||
setRequired() {
|
setRequired() {
|
||||||
@@ -163,7 +183,78 @@ patch(ListRenderer.prototype, 'jikimo_frontend.ListRenderer', {
|
|||||||
} catch (e) {
|
} catch (e) {
|
||||||
console.log(e)
|
console.log(e)
|
||||||
}
|
}
|
||||||
|
},
|
||||||
|
setColumns( merge_key) {
|
||||||
|
merge_key = merge_key.split(',')
|
||||||
|
const data = this.props.list.records
|
||||||
|
let sourceIndex = 0;
|
||||||
|
let sourceValue = merge_key.reduce((acc, key) => {
|
||||||
|
acc[key] = '';
|
||||||
|
return acc;
|
||||||
|
}, {});
|
||||||
|
|
||||||
|
data.forEach((item, index) => {
|
||||||
|
if(!item.colspan) {
|
||||||
|
item.colspan = 1;
|
||||||
|
}
|
||||||
|
const itemValue = merge_key.reduce((acc, key) => {
|
||||||
|
acc[key] = item.data[key];
|
||||||
|
return acc;
|
||||||
|
}, {});
|
||||||
|
if(JSON.stringify(itemValue) == JSON.stringify(sourceValue)) {
|
||||||
|
data[sourceIndex].colspan ++ ;
|
||||||
|
item.colspan = 0;
|
||||||
|
} else {
|
||||||
|
sourceIndex = index;
|
||||||
|
sourceValue = itemValue;
|
||||||
|
}
|
||||||
|
})
|
||||||
|
return data
|
||||||
|
},
|
||||||
|
getFieldModifiers(xmlString) {
|
||||||
|
const parser = new DOMParser();
|
||||||
|
const xmlDoc = parser.parseFromString(xmlString, "text/xml");
|
||||||
|
|
||||||
|
// 提取 <tree> 的 modifiers
|
||||||
|
const treeElement = xmlDoc.querySelector("tree");
|
||||||
|
const treeModifiers = treeElement.getAttribute("modifiers");
|
||||||
|
if (treeModifiers) {
|
||||||
|
const parsedTreeModifiers = JSON.parse(treeModifiers);
|
||||||
|
console.log("Tree Modifiers:", parsedTreeModifiers);
|
||||||
|
return parsedTreeModifiers;
|
||||||
|
}
|
||||||
|
return null;
|
||||||
|
},
|
||||||
|
mergeColumns(merge_fields, data) {
|
||||||
|
const dom = this.tableRef.el
|
||||||
|
const thead = $(dom).children('thead')
|
||||||
|
const tbody = $(dom).children('tbody')
|
||||||
|
let row_no = 0
|
||||||
|
tbody.children('tr.o_data_row').each(function () {
|
||||||
|
const tr = $(this)
|
||||||
|
const td = tr.children('td')
|
||||||
|
const index = $(this).index()
|
||||||
|
const col = data[index].colspan
|
||||||
|
row_no ++
|
||||||
|
if(col == 0) {
|
||||||
|
row_no --
|
||||||
|
}
|
||||||
|
td.eq(0).text(row_no).attr('rowspan', col)
|
||||||
|
|
||||||
|
if(col == 0) {
|
||||||
|
td.eq(0).remove()
|
||||||
|
}
|
||||||
|
td.each(function () {
|
||||||
|
if(merge_fields.indexOf($(this).attr('name')) >= 0) {
|
||||||
|
$(this).attr('rowspan', col)
|
||||||
|
if(col == 0) {
|
||||||
|
$(this).remove()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
})
|
||||||
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
})
|
})
|
||||||
patch(FormLabel.prototype, 'jikimo_frontend.FormLabel', {
|
patch(FormLabel.prototype, 'jikimo_frontend.FormLabel', {
|
||||||
get className() {
|
get className() {
|
||||||
@@ -176,7 +267,6 @@ patch(FormLabel.prototype, 'jikimo_frontend.FormLabel', {
|
|||||||
);
|
);
|
||||||
const classes = this.props.className ? [this.props.className] : [];
|
const classes = this.props.className ? [this.props.className] : [];
|
||||||
const otherRequired = filedRequiredList[this.props.fieldName]
|
const otherRequired = filedRequiredList[this.props.fieldName]
|
||||||
|
|
||||||
if(this.props.fieldInfo?.rawAttrs?.class?.indexOf('custom_required') >= 0 || otherRequired) {
|
if(this.props.fieldInfo?.rawAttrs?.class?.indexOf('custom_required') >= 0 || otherRequired) {
|
||||||
classes.push('custom_required_add')
|
classes.push('custom_required_add')
|
||||||
}
|
}
|
||||||
@@ -193,35 +283,6 @@ patch(FormLabel.prototype, 'jikimo_frontend.FormLabel', {
|
|||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
// 根据进度条设置水印
|
|
||||||
// const statusbar_params = {
|
|
||||||
// '已完工': 'bg-primary',
|
|
||||||
// '完成': 'bg-primary',
|
|
||||||
// '采购订单': 'bg-primary',
|
|
||||||
// '作废': 'bg-danger',
|
|
||||||
// '封存(报废)': 'bg-danger',
|
|
||||||
// }
|
|
||||||
// patch(StatusBarField.prototype, 'jikimo_frontend.StatusBarField', {
|
|
||||||
// setup() {
|
|
||||||
// owl.onMounted(this.ribbons);
|
|
||||||
// return this._super(...arguments);
|
|
||||||
// },
|
|
||||||
// ribbons() {
|
|
||||||
// try {
|
|
||||||
// const dom = $('.o_form_sheet.position-relative')
|
|
||||||
// const status = statusbar_params[this.currentName]
|
|
||||||
// if(status && dom.length) {
|
|
||||||
// dom.prepend(`<div class="o_widget o_widget_web_ribbon">
|
|
||||||
// <div class="ribbon ribbon-top-right">
|
|
||||||
// <span class="bg-opacity-75 ${status}" title="">${this.currentName}</span>
|
|
||||||
// </div>
|
|
||||||
// </div>`)
|
|
||||||
// }
|
|
||||||
// } catch (e) {
|
|
||||||
// console.log(e)
|
|
||||||
// }
|
|
||||||
// }
|
|
||||||
// })
|
|
||||||
|
|
||||||
$(function () {
|
$(function () {
|
||||||
|
|
||||||
|
|||||||
@@ -157,11 +157,11 @@ td.o_required_modifier {
|
|||||||
color: #aaa;
|
color: #aaa;
|
||||||
}
|
}
|
||||||
|
|
||||||
.o_kanban_primary_left {
|
// .o_kanban_primary_left {
|
||||||
display: flex;
|
// display: flex;
|
||||||
flex-direction: row-reverse;
|
// flex-direction: row-reverse;
|
||||||
justify-content: flex-start;
|
// justify-content: flex-start;
|
||||||
}
|
// }
|
||||||
|
|
||||||
.o_list_view .o_list_table thead {
|
.o_list_view .o_list_table thead {
|
||||||
position: sticky;
|
position: sticky;
|
||||||
@@ -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
|
||||||
|
|
||||||
31
jikimo_printing/models/workorder_printing.py
Normal file
31
jikimo_printing/models/workorder_printing.py
Normal file
@@ -0,0 +1,31 @@
|
|||||||
|
|
||||||
|
import logging
|
||||||
|
from odoo import models, fields, api
|
||||||
|
|
||||||
|
_logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
class MrpWorkorder(models.Model):
|
||||||
|
_name = 'mrp.workorder'
|
||||||
|
_inherit = ['mrp.workorder']
|
||||||
|
|
||||||
|
def _compute_state(self):
|
||||||
|
super(MrpWorkorder, self)._compute_state()
|
||||||
|
for workorder in self:
|
||||||
|
work_ids = workorder.production_id.workorder_ids.filtered(lambda w: w.routing_type == '装夹预调' or w.routing_type == '人工线下加工')
|
||||||
|
for wo in work_ids:
|
||||||
|
if wo.state == 'ready' and not wo.production_id.product_id.is_print_program:
|
||||||
|
# 触发打印程序
|
||||||
|
pdf_data = workorder.processing_drawing
|
||||||
|
if pdf_data:
|
||||||
|
try:
|
||||||
|
# 执行打印
|
||||||
|
self.env['jikimo.printing'].print_pdf(pdf_data)
|
||||||
|
wo.production_id.product_id.is_print_program = True
|
||||||
|
except Exception as e:
|
||||||
|
_logger.error(f"工单 {wo.name} 的PDF打印失败: {str(e)}")
|
||||||
|
|
||||||
|
class ProductTemplate(models.Model):
|
||||||
|
_inherit = 'product.template'
|
||||||
|
|
||||||
|
is_print_program = fields.Boolean(string='是否打印程序', default=False)
|
||||||
|
|
||||||
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>
|
||||||
|
|
||||||
|
|
||||||
3
jikimo_purchase_request/__init__.py
Normal file
3
jikimo_purchase_request/__init__.py
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
from . import models
|
||||||
|
from . import wizard
|
||||||
30
jikimo_purchase_request/__manifest__.py
Normal file
30
jikimo_purchase_request/__manifest__.py
Normal file
@@ -0,0 +1,30 @@
|
|||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
{
|
||||||
|
'name': '机企猫 采购申请',
|
||||||
|
'version': '16.0.1.0.0',
|
||||||
|
'summary': """ 机企猫 采购申请 """,
|
||||||
|
'author': '机企猫',
|
||||||
|
'website': 'https://bfw.jikimo.com',
|
||||||
|
'category': 'purchase',
|
||||||
|
'depends': ['sf_manufacturing', 'purchase_request'],
|
||||||
|
'data': [
|
||||||
|
'security/ir.model.access.csv',
|
||||||
|
'views/sale_order_view.xml',
|
||||||
|
'views/mrp_production.xml',
|
||||||
|
'views/purchase_request_view.xml',
|
||||||
|
'wizard/purchase_request_line_make_purchase_order_view.xml',
|
||||||
|
'views/purchase_request_line_view.xml',
|
||||||
|
'views/stock_picking_views.xml',
|
||||||
|
'wizard/purchase_request_wizard_views.xml',
|
||||||
|
'views/purchase_request_menu_views.xml',
|
||||||
|
],
|
||||||
|
'assets': {
|
||||||
|
'web.assets_backend': [
|
||||||
|
'jikimo_purchase_request/static/src/**/*'
|
||||||
|
],
|
||||||
|
},
|
||||||
|
'application': True,
|
||||||
|
'installable': True,
|
||||||
|
'auto_install': False,
|
||||||
|
'license': 'LGPL-3',
|
||||||
|
}
|
||||||
1573
jikimo_purchase_request/i18n/zh_CN.po
Normal file
1573
jikimo_purchase_request/i18n/zh_CN.po
Normal file
File diff suppressed because it is too large
Load Diff
9
jikimo_purchase_request/models/__init__.py
Normal file
9
jikimo_purchase_request/models/__init__.py
Normal file
@@ -0,0 +1,9 @@
|
|||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
from . import product_template
|
||||||
|
from . import purchase_request
|
||||||
|
from . import sale_order
|
||||||
|
from . import mrp_production
|
||||||
|
from . import purchase_order
|
||||||
|
from . import stock_rule
|
||||||
|
from . import stock_picking
|
||||||
|
from . import product_product
|
||||||
52
jikimo_purchase_request/models/mrp_production.py
Normal file
52
jikimo_purchase_request/models/mrp_production.py
Normal file
@@ -0,0 +1,52 @@
|
|||||||
|
from odoo import fields, models, api, _
|
||||||
|
|
||||||
|
|
||||||
|
class MrpProduction(models.Model):
|
||||||
|
_inherit = 'mrp.production'
|
||||||
|
|
||||||
|
pr_mp_count = fields.Integer('采购申请单数量', compute='_compute_pr_mp_count', store=True)
|
||||||
|
|
||||||
|
@api.depends('state')
|
||||||
|
def _compute_pr_mp_count(self):
|
||||||
|
for item in self:
|
||||||
|
if item.product_id.is_customer_provided:
|
||||||
|
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):
|
||||||
|
"""
|
||||||
|
采购请求
|
||||||
|
"""
|
||||||
|
self.ensure_one()
|
||||||
|
|
||||||
|
# 由于采购申请合并了所有销售订单行的采购,所以不区分产品
|
||||||
|
pr_ids = self._get_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': _("从 %s生成采购请求单", self.name),
|
||||||
|
'domain': [('id', 'in', pr_ids.ids)],
|
||||||
|
'view_mode': 'tree,form',
|
||||||
|
})
|
||||||
|
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
|
||||||
19
jikimo_purchase_request/models/product_template.py
Normal file
19
jikimo_purchase_request/models/product_template.py
Normal file
@@ -0,0 +1,19 @@
|
|||||||
|
from odoo import models, fields
|
||||||
|
|
||||||
|
|
||||||
|
class ProductTemplate(models.Model):
|
||||||
|
_inherit = 'product.template'
|
||||||
|
|
||||||
|
purchase_request_id = fields.Many2one('purchase.request', string='采购申请')
|
||||||
|
|
||||||
|
def no_bom_product_create(self, product_id, item, order_id, route_type, i, finish_product):
|
||||||
|
""" 创建坯料时,复制采购申请 """
|
||||||
|
template_id = super(ProductTemplate, self).no_bom_product_create(product_id, item, order_id, route_type, i,
|
||||||
|
finish_product)
|
||||||
|
template_id.purchase_request = product_id.purchase_request
|
||||||
|
return template_id
|
||||||
|
|
||||||
|
def copy_template(self, product_template_id):
|
||||||
|
""" 复制成品模板时,复制采购申请 """
|
||||||
|
super(ProductTemplate, self).copy_template(product_template_id)
|
||||||
|
self.purchase_request = product_template_id.purchase_request
|
||||||
56
jikimo_purchase_request/models/purchase_order.py
Normal file
56
jikimo_purchase_request/models/purchase_order.py
Normal file
@@ -0,0 +1,56 @@
|
|||||||
|
from odoo import api, fields, models, _
|
||||||
|
from odoo.tools import float_compare
|
||||||
|
|
||||||
|
|
||||||
|
class PurchaseOrder(models.Model):
|
||||||
|
_inherit = 'purchase.order'
|
||||||
|
|
||||||
|
state = fields.Selection([
|
||||||
|
('draft', '询价'),
|
||||||
|
('sent', '发送询价'),
|
||||||
|
('to approve', '待批准'),
|
||||||
|
("approved", "已批准"),
|
||||||
|
('purchase', '采购订单'),
|
||||||
|
('done', '完成'),
|
||||||
|
('cancel', '取消'),
|
||||||
|
('rejected', '已驳回')
|
||||||
|
], string='Status', readonly=True, index=True, copy=False, default='draft', tracking=True)
|
||||||
|
|
||||||
|
|
||||||
|
def button_confirm(self):
|
||||||
|
res = super(PurchaseOrder, self).button_confirm()
|
||||||
|
# 取消反向调拨单
|
||||||
|
reverse_move_ids = self.env['stock.move'].search([
|
||||||
|
('origin', '=', self.name),
|
||||||
|
('purchase_line_id', '=', False),
|
||||||
|
('state', '!=', 'done')
|
||||||
|
])
|
||||||
|
if reverse_move_ids:
|
||||||
|
reverse_move_ids.picking_id.action_cancel()
|
||||||
|
return res
|
||||||
|
|
||||||
|
def button_cancel(self):
|
||||||
|
"""
|
||||||
|
将取消的采购订单关联的库存移动撤销
|
||||||
|
"""
|
||||||
|
move_ids = self.order_line.move_dest_ids.filtered(lambda move: move.state != 'done' and not move.scrapped)
|
||||||
|
res =super(PurchaseOrder, self).button_cancel()
|
||||||
|
if move_ids.mapped('created_purchase_request_line_id'):
|
||||||
|
move_ids.write({'state': 'waiting', 'is_done': False})
|
||||||
|
return res
|
||||||
|
|
||||||
|
def write(self, vals):
|
||||||
|
res = super(PurchaseOrder, self).write(vals)
|
||||||
|
if 'state' in vals and vals['state'] == 'purchase':
|
||||||
|
purchase_request = self.order_line.purchase_request_lines.request_id
|
||||||
|
if purchase_request:
|
||||||
|
finished = True
|
||||||
|
# 判断该采购申请所有明细行是否都完成
|
||||||
|
for purchase_request_line in purchase_request.line_ids:
|
||||||
|
finished_qty = sum(purchase_request_line.purchase_lines.filtered(lambda line: line.state == 'purchase').mapped('product_qty'))
|
||||||
|
if float_compare(finished_qty ,purchase_request_line.product_qty, precision_rounding=purchase_request_line.product_id.uom_id.rounding) < 0:
|
||||||
|
finished = False
|
||||||
|
break
|
||||||
|
if finished:
|
||||||
|
purchase_request.button_done()
|
||||||
|
return res
|
||||||
202
jikimo_purchase_request/models/purchase_request.py
Normal file
202
jikimo_purchase_request/models/purchase_request.py
Normal file
@@ -0,0 +1,202 @@
|
|||||||
|
import re
|
||||||
|
import ast
|
||||||
|
from odoo import models, fields, api, _
|
||||||
|
from itertools import groupby
|
||||||
|
from odoo.tools import float_compare
|
||||||
|
|
||||||
|
|
||||||
|
class PurchaseRequest(models.Model):
|
||||||
|
_inherit = 'purchase.request'
|
||||||
|
_description = '采购申请'
|
||||||
|
|
||||||
|
# 为state添加取消状态
|
||||||
|
state = fields.Selection(
|
||||||
|
selection_add=[('cancel', '已取消')],
|
||||||
|
ondelete={'cancel': 'set default'} # 添加 ondelete 策略
|
||||||
|
)
|
||||||
|
|
||||||
|
rule_new_add = fields.Boolean('采购请求为规则创建', default=False, compute='_compute_state', store=True)
|
||||||
|
|
||||||
|
@api.depends('state')
|
||||||
|
def _compute_state(self):
|
||||||
|
for pr in self:
|
||||||
|
if pr.state != 'draft' and pr.rule_new_add:
|
||||||
|
pr.rule_new_add = False
|
||||||
|
|
||||||
|
def action_view_purchase_order(self):
|
||||||
|
action = super(PurchaseRequest, self).action_view_purchase_order()
|
||||||
|
origin_context = ast.literal_eval(action['context'])
|
||||||
|
if 'search_default_draft' in origin_context:
|
||||||
|
origin_context.pop('search_default_draft')
|
||||||
|
action['context'] = origin_context
|
||||||
|
return action
|
||||||
|
|
||||||
|
def button_done(self):
|
||||||
|
product_qty_map = {key: sum(line.product_qty for line in group) for key, group in
|
||||||
|
groupby(self.line_ids, key=lambda x: x.product_id.id)}
|
||||||
|
lines = self.mapped("line_ids.purchase_lines.order_id")
|
||||||
|
# 采购单产品和数量
|
||||||
|
product_summary = {}
|
||||||
|
product_rounding = {}
|
||||||
|
if lines:
|
||||||
|
for line in lines:
|
||||||
|
for line_item in line.order_line:
|
||||||
|
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):
|
||||||
|
_inherit = 'purchase.request.line'
|
||||||
|
_description = '采购申请明细'
|
||||||
|
|
||||||
|
origin = fields.Char(string="Source Document")
|
||||||
|
|
||||||
|
part_number = fields.Char('零件图号', store=True, compute='_compute_part_number')
|
||||||
|
part_name = fields.Char('零件名称', store=True, compute='_compute_part_number')
|
||||||
|
related_product = fields.Many2one('product.product', string='关联产品',
|
||||||
|
help='经此产品工艺加工成的成品')
|
||||||
|
|
||||||
|
supply_method = fields.Selection([
|
||||||
|
('automation', "自动化产线加工"),
|
||||||
|
('manual', "人工线下加工"),
|
||||||
|
('purchase', "外购"),
|
||||||
|
('outsourcing', "委外加工"),
|
||||||
|
], string='供货方式', compute='_compute_supply_method', store=True)
|
||||||
|
|
||||||
|
purchase_request_count = fields.Integer(string='采购申请数量', compute='_compute_purchase_request_count',
|
||||||
|
readonly=True)
|
||||||
|
purchase_count = fields.Integer(string="采购订单数量", compute="_compute_purchase_count", readonly=True)
|
||||||
|
|
||||||
|
@api.depends("purchase_lines")
|
||||||
|
def _compute_purchase_count(self):
|
||||||
|
for rec in self:
|
||||||
|
rec.purchase_count = len(rec.mapped("purchase_lines.order_id"))
|
||||||
|
|
||||||
|
@api.depends('request_id')
|
||||||
|
def _compute_purchase_request_count(self):
|
||||||
|
for order in self:
|
||||||
|
order.purchase_request_count = len(order.request_id)
|
||||||
|
|
||||||
|
@api.depends('origin')
|
||||||
|
def _compute_supply_method(self):
|
||||||
|
for prl in self:
|
||||||
|
order_ids = []
|
||||||
|
if not prl.origin:
|
||||||
|
continue
|
||||||
|
origin = [origin.replace(' ', '') for origin in prl.origin.split(',')]
|
||||||
|
if 'S' in prl.origin:
|
||||||
|
# 原单据是销售订单
|
||||||
|
order_ids = self.env['sale.order'].sudo().search([('name', 'in', origin)]).ids
|
||||||
|
elif 'MO' in prl.origin:
|
||||||
|
# 原单据是制造订单
|
||||||
|
mp_ids = self.env['mrp.production'].sudo().search([('name', 'in', origin)])
|
||||||
|
order_ids = [mp_id.sale_order_id.id for mp_id in mp_ids] if mp_ids else []
|
||||||
|
elif 'WH' in prl.origin:
|
||||||
|
# 原单据是调拨单
|
||||||
|
sp_ids = self.env['stock.picking'].sudo().search([('name', 'in', origin)])
|
||||||
|
order_ids = [sp_id.sale_order_id.id for sp_id in sp_ids] if sp_ids else []
|
||||||
|
order_line = self.env['sale.order.line'].sudo().search(
|
||||||
|
[('product_id', '=', prl.product_id.id), ('order_id', 'in', order_ids)])
|
||||||
|
if order_line:
|
||||||
|
prl.supply_method = order_line[0].supply_method
|
||||||
|
else:
|
||||||
|
prl.supply_method = None
|
||||||
|
|
||||||
|
@api.depends('product_id')
|
||||||
|
def _compute_part_number(self):
|
||||||
|
for record in self:
|
||||||
|
if record.part_number and record.part_name:
|
||||||
|
continue
|
||||||
|
if record.product_id.categ_id.name == '坯料':
|
||||||
|
product_name = ''
|
||||||
|
match = re.search(r'(S\d{5}-\d+)', record.product_id.name)
|
||||||
|
# 如果匹配成功,提取结果
|
||||||
|
if match:
|
||||||
|
product_name = match.group(0)
|
||||||
|
else:
|
||||||
|
product_name = record.product_id.name
|
||||||
|
sale_order_name = ''
|
||||||
|
match_sale = re.search(r'S(\d+)', record.product_id.name)
|
||||||
|
if match_sale:
|
||||||
|
sale_order_name = match_sale.group(0)
|
||||||
|
sale_order = self.env['sale.order'].sudo().search(
|
||||||
|
[('name', '=', sale_order_name)])
|
||||||
|
if sale_order:
|
||||||
|
filtered_order_line = sale_order.order_line.filtered(
|
||||||
|
lambda order_line: re.search(f'{product_name}$', order_line.product_id.name)
|
||||||
|
)
|
||||||
|
record.part_number = filtered_order_line.product_id.part_number
|
||||||
|
record.part_name = filtered_order_line.product_id.part_name
|
||||||
|
else:
|
||||||
|
record.part_number = record.product_id.part_number
|
||||||
|
record.part_name = record.product_id.part_name
|
||||||
|
|
||||||
|
def _compute_qty_to_buy(self):
|
||||||
|
for pr in self:
|
||||||
|
qty_to_buy = sum(pr.mapped("product_qty"))
|
||||||
|
if pr.purchase_count > 0:
|
||||||
|
qty_to_buy -= sum(pr.mapped("purchase_lines").filtered(lambda po: po.state != 'cancel').mapped(
|
||||||
|
"product_qty"))
|
||||||
|
pr.qty_to_buy = qty_to_buy > 0.0
|
||||||
|
pr.pending_qty_to_receive = qty_to_buy
|
||||||
|
|
||||||
|
def action_view_purchase_request(self):
|
||||||
|
action = self.env["ir.actions.actions"]._for_xml_id("purchase_request.purchase_request_form_action")
|
||||||
|
action.update({
|
||||||
|
'res_id': self.request_id.id,
|
||||||
|
'views': [[False, 'form']],
|
||||||
|
})
|
||||||
|
return action
|
||||||
|
|
||||||
|
def action_view_purchase_order(self):
|
||||||
|
action = self.env["ir.actions.actions"]._for_xml_id("purchase.purchase_rfq")
|
||||||
|
lines = self.mapped("purchase_lines.order_id")
|
||||||
|
if len(lines) > 1:
|
||||||
|
action["domain"] = [("id", "in", lines.ids)]
|
||||||
|
elif lines:
|
||||||
|
action["views"] = [
|
||||||
|
(self.env.ref("purchase.purchase_order_form").id, "form")
|
||||||
|
]
|
||||||
|
action["res_id"] = lines.id
|
||||||
|
origin_context = ast.literal_eval(action['context'])
|
||||||
|
if 'search_default_draft' in origin_context:
|
||||||
|
origin_context.pop('search_default_draft')
|
||||||
|
action['context'] = origin_context
|
||||||
|
return action
|
||||||
50
jikimo_purchase_request/models/sale_order.py
Normal file
50
jikimo_purchase_request/models/sale_order.py
Normal file
@@ -0,0 +1,50 @@
|
|||||||
|
from odoo import fields, models, api, _
|
||||||
|
|
||||||
|
|
||||||
|
class StatusChange(models.Model):
|
||||||
|
_inherit = 'sale.order'
|
||||||
|
|
||||||
|
# def action_confirm(self):
|
||||||
|
# res = super(StatusChange, self).action_confirm()
|
||||||
|
# # 采购申请自动确认
|
||||||
|
# pr_ids = self.env["purchase.request"].sudo().search(
|
||||||
|
# [('origin', 'like', self.name), ('rule_new_add', '=', True)])
|
||||||
|
# if pr_ids:
|
||||||
|
# pr_ids.write({'need_validation': False})
|
||||||
|
# pr_ids.write({"state": "approved"})
|
||||||
|
# return res
|
||||||
|
|
||||||
|
purchase_request_purchase_order_count = fields.Integer('采购申请单数量', compute='_compute_purchase_request_count',
|
||||||
|
store=True)
|
||||||
|
|
||||||
|
@api.depends('state')
|
||||||
|
def _compute_purchase_request_count(self):
|
||||||
|
for so in self:
|
||||||
|
pr_ids = self.env['purchase.request'].sudo().search([('origin', 'like', so.name)])
|
||||||
|
if pr_ids:
|
||||||
|
so.purchase_request_purchase_order_count = len(pr_ids)
|
||||||
|
else:
|
||||||
|
so.purchase_request_purchase_order_count = 0
|
||||||
|
|
||||||
|
def action_view_purchase_request_purchase_orders(self):
|
||||||
|
"""
|
||||||
|
采购请求
|
||||||
|
"""
|
||||||
|
self.ensure_one()
|
||||||
|
pr_ids = self.env['purchase.request'].sudo().search([('origin', 'like', self.name)])
|
||||||
|
action = {
|
||||||
|
'res_model': 'purchase.request',
|
||||||
|
'type': 'ir.actions.act_window',
|
||||||
|
}
|
||||||
|
if len(pr_ids) == 1:
|
||||||
|
action.update({
|
||||||
|
'view_mode': 'form',
|
||||||
|
'res_id': pr_ids[0].id,
|
||||||
|
})
|
||||||
|
else:
|
||||||
|
action.update({
|
||||||
|
'name': _("从 %s生成采购请求单", self.name),
|
||||||
|
'domain': [('id', 'in', pr_ids)],
|
||||||
|
'view_mode': 'tree,form',
|
||||||
|
})
|
||||||
|
return action
|
||||||
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
|
||||||
82
jikimo_purchase_request/models/stock_rule.py
Normal file
82
jikimo_purchase_request/models/stock_rule.py
Normal file
@@ -0,0 +1,82 @@
|
|||||||
|
from odoo import api, fields, models
|
||||||
|
from collections import defaultdict
|
||||||
|
|
||||||
|
|
||||||
|
class StockRule(models.Model):
|
||||||
|
_inherit = "stock.rule"
|
||||||
|
|
||||||
|
def create_purchase_request(self, procurement_group):
|
||||||
|
"""
|
||||||
|
Create a purchase request containing procurement order product.
|
||||||
|
"""
|
||||||
|
procurement = procurement_group[0]
|
||||||
|
rule = procurement_group[1]
|
||||||
|
purchase_request_model = self.env["purchase.request"]
|
||||||
|
purchase_request_line_model = self.env["purchase.request.line"]
|
||||||
|
cache = {}
|
||||||
|
pr = self.env["purchase.request"]
|
||||||
|
domain = rule._make_pr_get_domain(procurement.values)
|
||||||
|
if domain in cache:
|
||||||
|
pr = cache[domain]
|
||||||
|
elif domain:
|
||||||
|
pr = self.env["purchase.request"].search([dom for dom in domain])
|
||||||
|
pr = pr[0] if pr else False
|
||||||
|
cache[domain] = pr
|
||||||
|
if not pr:
|
||||||
|
request_data = rule._prepare_purchase_request(
|
||||||
|
procurement.origin, procurement.values
|
||||||
|
)
|
||||||
|
request_data.update({'rule_new_add': True})
|
||||||
|
pr = purchase_request_model.create(request_data)
|
||||||
|
cache[domain] = pr
|
||||||
|
elif (
|
||||||
|
not pr.origin
|
||||||
|
or procurement.origin not in pr.origin.split(", ")
|
||||||
|
and procurement.origin != "/"
|
||||||
|
):
|
||||||
|
if pr.origin:
|
||||||
|
if procurement.origin:
|
||||||
|
pr.write({"origin": pr.origin + ", " + procurement.origin})
|
||||||
|
else:
|
||||||
|
pr.write({"origin": procurement.origin})
|
||||||
|
# Create Line
|
||||||
|
request_line_data = rule._prepare_purchase_request_line(pr, procurement)
|
||||||
|
request_line_data.update({'origin': procurement.origin})
|
||||||
|
purchase_request_line_model.create(request_line_data)
|
||||||
|
|
||||||
|
def _run_buy(self, procurements):
|
||||||
|
# 如果补货组相同,并且产品相同,则合并
|
||||||
|
procurements_dict = defaultdict()
|
||||||
|
for procurement, rule in procurements:
|
||||||
|
if (procurement.product_id.id, procurement.values['group_id'], rule.id) not in procurements_dict:
|
||||||
|
procurements_dict[(procurement.product_id.id, procurement.values['group_id'], rule.id)] = {
|
||||||
|
'product_id': procurement.product_id,
|
||||||
|
'product_qty': procurement.product_qty,
|
||||||
|
'product_uom': procurement.product_uom,
|
||||||
|
'location_id': procurement.location_id,
|
||||||
|
'name': procurement.name,
|
||||||
|
'origin': procurement.origin,
|
||||||
|
'company_id': procurement.company_id,
|
||||||
|
'values': procurement.values,
|
||||||
|
'rule': rule
|
||||||
|
}
|
||||||
|
else:
|
||||||
|
procurements_dict[(procurement.product_id.id, procurement.values['group_id'], rule.id)]['product_qty'] += procurement.product_qty
|
||||||
|
procurements_dict[(procurement.product_id.id, procurement.values['group_id'], rule.id)]['values']['move_dest_ids'] |= procurement.values['move_dest_ids']
|
||||||
|
new_procurements = []
|
||||||
|
for k, p in procurements_dict.items():
|
||||||
|
new_procurements.append((
|
||||||
|
self.env['procurement.group'].Procurement(
|
||||||
|
product_id=p['product_id'],
|
||||||
|
product_qty=p['product_qty'],
|
||||||
|
product_uom=p['product_uom'],
|
||||||
|
location_id=p['location_id'],
|
||||||
|
name=p['name'],
|
||||||
|
origin=p['origin'],
|
||||||
|
company_id=p['company_id'],
|
||||||
|
values=p['values']
|
||||||
|
), p['rule'])
|
||||||
|
)
|
||||||
|
|
||||||
|
res = super(StockRule, self)._run_buy(new_procurements)
|
||||||
|
return res
|
||||||
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;
|
||||||
|
}
|
||||||
21
jikimo_purchase_request/views/mrp_production.xml
Normal file
21
jikimo_purchase_request/views/mrp_production.xml
Normal file
@@ -0,0 +1,21 @@
|
|||||||
|
<?xml version="1.0" encoding="UTF-8" ?>
|
||||||
|
<odoo>
|
||||||
|
<record id="mrp_production_inherited_form_purchase_request" model="ir.ui.view">
|
||||||
|
<field name="name">mrp.production.inherited.form.purchase.request</field>
|
||||||
|
<field name="model">mrp.production</field>
|
||||||
|
<field name="inherit_id" ref="mrp.mrp_production_form_view"/>
|
||||||
|
<field name="arch" type="xml">
|
||||||
|
<xpath expr="//button[@name='action_view_mo_delivery']" position="before">
|
||||||
|
<button class="oe_stat_button" name="action_view_pr_mp" type="object" icon="fa-credit-card"
|
||||||
|
attrs="{'invisible': [('pr_mp_count', '=', 0)]}">
|
||||||
|
<div class="o_field_widget o_stat_info">
|
||||||
|
<span class="o_stat_value">
|
||||||
|
<field name="pr_mp_count"/>
|
||||||
|
</span>
|
||||||
|
<span class="o_stat_text">采购申请</span>
|
||||||
|
</div>
|
||||||
|
</button>
|
||||||
|
</xpath>
|
||||||
|
</field>
|
||||||
|
</record>
|
||||||
|
</odoo>
|
||||||
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>
|
||||||
|
|
||||||
91
jikimo_purchase_request/views/purchase_request_view.xml
Normal file
91
jikimo_purchase_request/views/purchase_request_view.xml
Normal file
@@ -0,0 +1,91 @@
|
|||||||
|
<odoo>
|
||||||
|
<record id="view_purchase_request_form_sf" model="ir.ui.view">
|
||||||
|
<field name="name">purchase.request.sf.form</field>
|
||||||
|
<field name="model">purchase.request</field>
|
||||||
|
<field name="inherit_id" ref="purchase_request.view_purchase_request_form"/>
|
||||||
|
<field name="arch" type="xml">
|
||||||
|
<xpath expr="//button[@name='button_draft']" position="attributes">
|
||||||
|
<attribute name="string">重置草稿</attribute>
|
||||||
|
</xpath>
|
||||||
|
<xpath expr="//field[@name='line_ids']//field[@name='purchased_qty']" position="after">
|
||||||
|
<field name="supply_method"/>
|
||||||
|
</xpath>
|
||||||
|
<xpath expr="//field[@name='line_ids']//field[@name='name']" position="replace">
|
||||||
|
<field name="related_product"/>
|
||||||
|
<field name="part_number"/>
|
||||||
|
<field name="part_name"/>
|
||||||
|
</xpath>
|
||||||
|
<xpath expr="//button[@name='button_done']" position="attributes">
|
||||||
|
<attribute name="class"/>
|
||||||
|
</xpath>
|
||||||
|
<xpath expr="//button[@name='button_in_progress']" position="attributes">
|
||||||
|
<attribute name="invisible">1</attribute>
|
||||||
|
</xpath>
|
||||||
|
<xpath expr="//button[@name='%(purchase_request.action_purchase_request_line_make_purchase_order)d']" position="attributes">
|
||||||
|
<attribute name="class">oe_highlight</attribute>
|
||||||
|
</xpath>
|
||||||
|
</field>
|
||||||
|
</record>
|
||||||
|
|
||||||
|
<record id="view_purchase_request_tree_sf" model="ir.ui.view">
|
||||||
|
<field name="name">purchase.request.sf.tree</field>
|
||||||
|
<field name="model">purchase.request</field>
|
||||||
|
<field name="inherit_id" ref="purchase_request.view_purchase_request_form"/>
|
||||||
|
<field name="arch" type="xml">
|
||||||
|
<xpath expr="//field[@name='activity_ids']" position="attributes">
|
||||||
|
<attribute name="optional">hide</attribute>
|
||||||
|
</xpath>
|
||||||
|
</field>
|
||||||
|
</record>
|
||||||
|
|
||||||
|
<record id="view_purchase_request_line_tree_sf" model="ir.ui.view">
|
||||||
|
<field name="name">purchase.request.line.sf.tree</field>
|
||||||
|
<field name="model">purchase.request.line</field>
|
||||||
|
<field name="inherit_id" ref="purchase_request.purchase_request_line_tree"/>
|
||||||
|
<field name="arch" type="xml">
|
||||||
|
<xpath expr="//field[@name='requested_by']" position="replace">
|
||||||
|
<field name="supply_method"/>
|
||||||
|
</xpath>
|
||||||
|
<xpath expr="//field[@name='assigned_to']" position="attributes">
|
||||||
|
<attribute name="invisible">True</attribute>
|
||||||
|
</xpath>
|
||||||
|
<xpath expr="//field[@name='name']" position="attributes">
|
||||||
|
<attribute name="invisible">True</attribute>
|
||||||
|
</xpath>
|
||||||
|
<xpath expr="//field[@name='supplier_id']" position="after">
|
||||||
|
<field name="requested_by" widget="many2one_avatar_user"/>
|
||||||
|
<field name="assigned_to" widget="many2one_avatar_user" invisible="1"/>
|
||||||
|
</xpath>
|
||||||
|
<xpath expr="//field[@name='purchased_qty']" position="attributes">
|
||||||
|
<attribute name="string">采购数量</attribute>
|
||||||
|
</xpath>
|
||||||
|
<xpath expr="//field[@name='purchase_state']" position="attributes">
|
||||||
|
<attribute name="string">订单状态</attribute>
|
||||||
|
</xpath>
|
||||||
|
<xpath expr="//field[@name='product_id']" position="after">
|
||||||
|
<field name="related_product"/>
|
||||||
|
<field name="part_number"/>
|
||||||
|
<field name="part_name" invisible="1"/>
|
||||||
|
</xpath>
|
||||||
|
</field>
|
||||||
|
</record>
|
||||||
|
|
||||||
|
<record id="view_purchase_request_line_search_sf" model="ir.ui.view">
|
||||||
|
<field name="name">purchase.request.line.sf.search</field>
|
||||||
|
<field name="model">purchase.request.line</field>
|
||||||
|
<field name="inherit_id" ref="purchase_request.purchase_request_line_search"/>
|
||||||
|
<field name="arch" type="xml">
|
||||||
|
<xpath expr="//field[@name='product_id']" position="after">
|
||||||
|
<field name="supply_method"/>
|
||||||
|
<field name="related_product"/>
|
||||||
|
<field name="part_number"/>
|
||||||
|
<field name="part_name"/>
|
||||||
|
</xpath>
|
||||||
|
</field>
|
||||||
|
</record>
|
||||||
|
|
||||||
|
<record model="ir.actions.act_window" id="purchase_request.purchase_request_form_action">
|
||||||
|
<field name="name">Purchase Requests</field>
|
||||||
|
<field name="context"></field>
|
||||||
|
</record>
|
||||||
|
</odoo>
|
||||||
19
jikimo_purchase_request/views/sale_order_view.xml
Normal file
19
jikimo_purchase_request/views/sale_order_view.xml
Normal file
@@ -0,0 +1,19 @@
|
|||||||
|
<odoo>
|
||||||
|
<record id="sale_order_inherited_form_purchase_request_sf" model="ir.ui.view">
|
||||||
|
<field name="name">sale.order.inherited.form.purchase.request</field>
|
||||||
|
<field name="model">sale.order</field>
|
||||||
|
<field name="inherit_id" ref="sale_purchase.sale_order_inherited_form_purchase"/>
|
||||||
|
<field name="arch" type="xml">
|
||||||
|
<xpath expr="//button[@name='action_preview_sale_order']" position="before">
|
||||||
|
<button class="oe_stat_button" name="action_view_purchase_request_purchase_orders" type="object" icon="fa-credit-card"
|
||||||
|
groups='purchase.group_purchase_user'
|
||||||
|
attrs="{'invisible': [('purchase_request_purchase_order_count', '=', 0)]}">
|
||||||
|
<div class="o_field_widget o_stat_info">
|
||||||
|
<span class="o_stat_value"><field name="purchase_request_purchase_order_count"/></span>
|
||||||
|
<span class="o_stat_text">采购申请</span>
|
||||||
|
</div>
|
||||||
|
</button>
|
||||||
|
</xpath>
|
||||||
|
</field>
|
||||||
|
</record>
|
||||||
|
</odoo>
|
||||||
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>
|
||||||
4
jikimo_purchase_request/wizard/__init__.py
Normal file
4
jikimo_purchase_request/wizard/__init__.py
Normal file
@@ -0,0 +1,4 @@
|
|||||||
|
# License LGPL-3.0 or later (https://www.gnu.org/licenses/lgpl-3.0)
|
||||||
|
|
||||||
|
from . import purchase_request_line_make_purchase_order
|
||||||
|
from . import purchase_request_wizard
|
||||||
@@ -0,0 +1,129 @@
|
|||||||
|
# Copyright 2018-2019 ForgeFlow, S.L.
|
||||||
|
# License LGPL-3.0 or later (https://www.gnu.org/licenses/lgpl-3.0).
|
||||||
|
from datetime import datetime
|
||||||
|
|
||||||
|
from odoo import _, api, fields, models
|
||||||
|
from odoo.exceptions import UserError, ValidationError
|
||||||
|
from odoo.tools import get_lang
|
||||||
|
|
||||||
|
|
||||||
|
class PurchaseRequestLineMakePurchaseOrder(models.TransientModel):
|
||||||
|
_inherit = "purchase.request.line.make.purchase.order"
|
||||||
|
|
||||||
|
def make_purchase_order(self):
|
||||||
|
res = []
|
||||||
|
purchase_obj = self.env["purchase.order"]
|
||||||
|
po_line_obj = self.env["purchase.order.line"]
|
||||||
|
purchase = False
|
||||||
|
|
||||||
|
if len(set([item_id.line_id.supply_method for item_id in self.item_ids])) > 1:
|
||||||
|
raise ValidationError('不同供货方式不可合并创建询价单!')
|
||||||
|
|
||||||
|
for item in self.item_ids:
|
||||||
|
line = item.line_id
|
||||||
|
if item.product_qty <= 0.0:
|
||||||
|
raise UserError(_("Enter a positive quantity."))
|
||||||
|
if self.purchase_order_id:
|
||||||
|
purchase = self.purchase_order_id
|
||||||
|
if not purchase:
|
||||||
|
po_data = self._prepare_purchase_order(
|
||||||
|
line.request_id.picking_type_id,
|
||||||
|
line.request_id.group_id,
|
||||||
|
line.company_id,
|
||||||
|
line.request_id.origin,
|
||||||
|
)
|
||||||
|
# po_data.update({'related_product':line.related_product.id})
|
||||||
|
purchase = purchase_obj.create(po_data)
|
||||||
|
|
||||||
|
# Look for any other PO line in the selected PO with same
|
||||||
|
# product and UoM to sum quantities instead of creating a new
|
||||||
|
# po line
|
||||||
|
domain = self._get_order_line_search_domain(purchase, item)
|
||||||
|
available_po_lines = po_line_obj.search(domain)
|
||||||
|
new_pr_line = True
|
||||||
|
# If Unit of Measure is not set, update from wizard.
|
||||||
|
if not line.product_uom_id:
|
||||||
|
line.product_uom_id = item.product_uom_id
|
||||||
|
# Allocation UoM has to be the same as PR line UoM
|
||||||
|
alloc_uom = line.product_uom_id
|
||||||
|
wizard_uom = item.product_uom_id
|
||||||
|
if available_po_lines and not item.keep_description:
|
||||||
|
new_pr_line = False
|
||||||
|
po_line = available_po_lines[0]
|
||||||
|
po_line.purchase_request_lines = [(4, line.id)]
|
||||||
|
po_line.move_dest_ids |= line.move_dest_ids
|
||||||
|
po_line_product_uom_qty = po_line.product_uom._compute_quantity(
|
||||||
|
po_line.product_uom_qty, alloc_uom
|
||||||
|
)
|
||||||
|
wizard_product_uom_qty = wizard_uom._compute_quantity(
|
||||||
|
item.product_qty, alloc_uom
|
||||||
|
)
|
||||||
|
all_qty = min(po_line_product_uom_qty, wizard_product_uom_qty)
|
||||||
|
self.create_allocation(po_line, line, all_qty, alloc_uom)
|
||||||
|
else:
|
||||||
|
po_line_data = self._prepare_purchase_order_line(purchase, item)
|
||||||
|
if item.keep_description:
|
||||||
|
po_line_data["name"] = item.name
|
||||||
|
if line.related_product:
|
||||||
|
po_line_data.update({'related_product': line.related_product.id})
|
||||||
|
po_line = po_line_obj.create(po_line_data)
|
||||||
|
po_line_product_uom_qty = po_line.product_uom._compute_quantity(
|
||||||
|
po_line.product_uom_qty, alloc_uom
|
||||||
|
)
|
||||||
|
wizard_product_uom_qty = wizard_uom._compute_quantity(
|
||||||
|
item.product_qty, alloc_uom
|
||||||
|
)
|
||||||
|
all_qty = min(po_line_product_uom_qty, wizard_product_uom_qty)
|
||||||
|
self.create_allocation(po_line, line, all_qty, alloc_uom)
|
||||||
|
self._post_process_po_line(item, po_line, new_pr_line)
|
||||||
|
res.append(purchase.id)
|
||||||
|
|
||||||
|
purchase_requests = self.item_ids.mapped("request_id")
|
||||||
|
purchase_requests.button_in_progress()
|
||||||
|
return {
|
||||||
|
"domain": [("id", "in", res)],
|
||||||
|
"name": _("RFQ"),
|
||||||
|
"view_mode": "tree,form",
|
||||||
|
"res_model": "purchase.order",
|
||||||
|
"view_id": False,
|
||||||
|
"context": False,
|
||||||
|
"type": "ir.actions.act_window",
|
||||||
|
}
|
||||||
|
|
||||||
|
def _check_valid_request_line(self, request_line_ids):
|
||||||
|
for line in self.env["purchase.request.line"].browse(request_line_ids):
|
||||||
|
if line.request_id.state not in ["approved", "in_progress"]:
|
||||||
|
raise UserError(
|
||||||
|
_("采购申请 %s 未审批或未进行中")
|
||||||
|
% line.request_id.name
|
||||||
|
)
|
||||||
|
super(PurchaseRequestLineMakePurchaseOrder, self)._check_valid_request_line(request_line_ids)
|
||||||
|
|
||||||
|
@api.model
|
||||||
|
def check_group(self, request_lines):
|
||||||
|
# 去掉合并必须同一采购组的限制
|
||||||
|
pass
|
||||||
|
|
||||||
|
def get_items(self, request_line_ids):
|
||||||
|
request_line_obj = self.env["purchase.request.line"]
|
||||||
|
items = []
|
||||||
|
request_lines = request_line_obj.browse(request_line_ids).filtered(lambda line: line.pending_qty_to_receive > 0)
|
||||||
|
self._check_valid_request_line(request_line_ids)
|
||||||
|
self.check_group(request_lines)
|
||||||
|
for line in request_lines:
|
||||||
|
items.append([0, 0, self._prepare_item(line)])
|
||||||
|
return items
|
||||||
|
|
||||||
|
|
||||||
|
class PurchaseRequestLineMakePurchaseOrderItem(models.TransientModel):
|
||||||
|
_inherit = "purchase.request.line.make.purchase.order.item"
|
||||||
|
|
||||||
|
supply_method = fields.Selection(related='line_id.supply_method', string='供货方式')
|
||||||
|
|
||||||
|
wiz_id = fields.Many2one(
|
||||||
|
comodel_name="purchase.request.line.make.purchase.order",
|
||||||
|
string="Wizard",
|
||||||
|
required=False,
|
||||||
|
ondelete="cascade",
|
||||||
|
readonly=True,
|
||||||
|
)
|
||||||
@@ -0,0 +1,13 @@
|
|||||||
|
<?xml version="1.0" encoding="utf-8" ?>
|
||||||
|
<odoo>
|
||||||
|
<record id="view_purchase_request_line_make_purchase_order_sf" model="ir.ui.view">
|
||||||
|
<field name="name">Purchase Request Line Make Purchase Order sf</field>
|
||||||
|
<field name="model">purchase.request.line.make.purchase.order</field>
|
||||||
|
<field name="inherit_id" ref="purchase_request.view_purchase_request_line_make_purchase_order"/>
|
||||||
|
<field name="arch" type="xml">
|
||||||
|
<xpath expr="//field[@name='item_ids']//field[@name='keep_description']" position="before">
|
||||||
|
<field name="supply_method"/>
|
||||||
|
</xpath>
|
||||||
|
</field>
|
||||||
|
</record>
|
||||||
|
</odoo>
|
||||||
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>
|
||||||
3
jikimo_purchase_request_tier_validation/__init__.py
Normal file
3
jikimo_purchase_request_tier_validation/__init__.py
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
|
||||||
|
from . import models
|
||||||
27
jikimo_purchase_request_tier_validation/__manifest__.py
Normal file
27
jikimo_purchase_request_tier_validation/__manifest__.py
Normal file
@@ -0,0 +1,27 @@
|
|||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
{
|
||||||
|
'name': "机企猫 采购申请审批流程",
|
||||||
|
|
||||||
|
'summary': """
|
||||||
|
采购申请审批流程""",
|
||||||
|
|
||||||
|
'description': """
|
||||||
|
Long description of module's purpose
|
||||||
|
""",
|
||||||
|
|
||||||
|
'author': "My Company",
|
||||||
|
'website': "https://www.yourcompany.com",
|
||||||
|
|
||||||
|
# Categories can be used to filter modules in modules listing
|
||||||
|
# Check https://github.com/odoo/odoo/blob/16.0/odoo/addons/base/data/ir_module_category_data.xml
|
||||||
|
# for the full list
|
||||||
|
'category': 'Uncategorized',
|
||||||
|
'version': '0.1',
|
||||||
|
|
||||||
|
# any module necessary for this one to work correctly
|
||||||
|
'depends': ['purchase_request_tier_validation'],
|
||||||
|
|
||||||
|
# always loaded
|
||||||
|
'data': [
|
||||||
|
],
|
||||||
|
}
|
||||||
@@ -0,0 +1,4 @@
|
|||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
|
||||||
|
from . import models
|
||||||
|
from . import stock_rule
|
||||||
30
jikimo_purchase_request_tier_validation/models/models.py
Normal file
30
jikimo_purchase_request_tier_validation/models/models.py
Normal file
@@ -0,0 +1,30 @@
|
|||||||
|
from odoo import models, fields, api, _
|
||||||
|
from odoo.exceptions import ValidationError
|
||||||
|
import logging
|
||||||
|
|
||||||
|
_logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
|
class PurchaseRequest(models.Model):
|
||||||
|
_inherit = 'purchase.request'
|
||||||
|
|
||||||
|
|
||||||
|
def _validate_tier(self, tiers=False):
|
||||||
|
res = super(PurchaseRequest, self)._validate_tier(tiers)
|
||||||
|
|
||||||
|
# 检查是否所有审批都已通过
|
||||||
|
all_approved = all(
|
||||||
|
tier_review.status == 'approved'
|
||||||
|
for tier_review in self.review_ids
|
||||||
|
)
|
||||||
|
|
||||||
|
if self.review_ids and all_approved: # 确保有审批记录
|
||||||
|
self.state = 'approved'
|
||||||
|
|
||||||
|
return res
|
||||||
|
|
||||||
|
@api.model
|
||||||
|
def _get_under_validation_exceptions(self):
|
||||||
|
res = super(PurchaseRequest, self)._get_under_validation_exceptions()
|
||||||
|
res.append("state")
|
||||||
|
return res
|
||||||
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
|
||||||
@@ -3,12 +3,10 @@
|
|||||||
'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
|
采购审批流程""",
|
||||||
""",
|
|
||||||
|
|
||||||
'author': "My Company",
|
'author': "My Company",
|
||||||
'website': "https://www.yourcompany.com",
|
'website': "https://www.yourcompany.com",
|
||||||
|
|||||||
17472
jikimo_purchase_tier_validation/i18n/zh_CN.po
Normal file
17472
jikimo_purchase_tier_validation/i18n/zh_CN.po
Normal file
File diff suppressed because it is too large
Load Diff
@@ -9,6 +9,8 @@ class jikimo_purchase_tier_validation(models.Model):
|
|||||||
_name = 'purchase.order'
|
_name = 'purchase.order'
|
||||||
_inherit = ['purchase.order', 'tier.validation']
|
_inherit = ['purchase.order', 'tier.validation']
|
||||||
_description = "采购订单"
|
_description = "采购订单"
|
||||||
|
_state_from = ["draft", "to approve", "rejected"]
|
||||||
|
_state_to = ["approved", "purchase"]
|
||||||
|
|
||||||
_tier_validation_buttons_xpath = "/form/header/button[@id='draft_confirm'][1]"
|
_tier_validation_buttons_xpath = "/form/header/button[@id='draft_confirm'][1]"
|
||||||
|
|
||||||
@@ -20,9 +22,9 @@ class jikimo_purchase_tier_validation(models.Model):
|
|||||||
is_upload_contract_file = fields.Boolean(string='是否已上传合同文件', default=False)
|
is_upload_contract_file = fields.Boolean(string='是否已上传合同文件', default=False)
|
||||||
|
|
||||||
def button_confirm(self):
|
def button_confirm(self):
|
||||||
for record in self:
|
# for record in self:
|
||||||
if record.state in ['to approve']:
|
# if record.need_validation and not record.validation_status == 'validated':
|
||||||
raise ValidationError(_('请先完成审批。'))
|
# raise ValidationError(_('请先完成审批。'))
|
||||||
res = super(jikimo_purchase_tier_validation, self).button_confirm()
|
res = super(jikimo_purchase_tier_validation, self).button_confirm()
|
||||||
for record in self:
|
for record in self:
|
||||||
if record.state == 'approved':
|
if record.state == 'approved':
|
||||||
@@ -68,11 +70,6 @@ class jikimo_purchase_tier_validation(models.Model):
|
|||||||
|
|
||||||
return res
|
return res
|
||||||
|
|
||||||
def _rejected_tier(self, tiers=False):
|
|
||||||
res = super(jikimo_purchase_tier_validation, self)._rejected_tier(tiers)
|
|
||||||
self.state = 'draft'
|
|
||||||
return res
|
|
||||||
|
|
||||||
@api.model
|
@api.model
|
||||||
def _get_under_validation_exceptions(self):
|
def _get_under_validation_exceptions(self):
|
||||||
res = super(jikimo_purchase_tier_validation, self)._get_under_validation_exceptions()
|
res = super(jikimo_purchase_tier_validation, self)._get_under_validation_exceptions()
|
||||||
|
|||||||
@@ -9,5 +9,6 @@ class MrpBom(models.Model):
|
|||||||
|
|
||||||
# 成品的供应商从模板中获取
|
# 成品的供应商从模板中获取
|
||||||
if product_type == 'product':
|
if product_type == 'product':
|
||||||
bom_id.subcontractor_id = product.product_tmpl_id.seller_ids.partner_id.id
|
if product.product_tmpl_id.seller_ids:
|
||||||
|
bom_id.subcontractor_id = product.product_tmpl_id.seller_ids[-1].partner_id.id
|
||||||
return bom_id
|
return bom_id
|
||||||
|
|||||||
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
|
||||||
|
|
||||||
3
jikimo_work_reporting_api/__init__.py
Normal file
3
jikimo_work_reporting_api/__init__.py
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
from . import controllers
|
||||||
|
from . import models
|
||||||
17
jikimo_work_reporting_api/__manifest__.py
Normal file
17
jikimo_work_reporting_api/__manifest__.py
Normal file
@@ -0,0 +1,17 @@
|
|||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
{
|
||||||
|
'name': '机企猫 报工系统API',
|
||||||
|
'version': '1.0.0',
|
||||||
|
'summary': """ 机企猫 报工系统API """,
|
||||||
|
'author': '机企猫',
|
||||||
|
'website': 'https://xt.sf.jikimo.com',
|
||||||
|
'category': 'sf',
|
||||||
|
'depends': ['base', 'sf_maintenance', 'jikimo_mini_program'],
|
||||||
|
'data': [
|
||||||
|
],
|
||||||
|
|
||||||
|
'application': True,
|
||||||
|
'installable': True,
|
||||||
|
'auto_install': False,
|
||||||
|
'license': 'LGPL-3',
|
||||||
|
}
|
||||||
2
jikimo_work_reporting_api/controllers/__init__.py
Normal file
2
jikimo_work_reporting_api/controllers/__init__.py
Normal file
@@ -0,0 +1,2 @@
|
|||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
from . import main
|
||||||
69
jikimo_work_reporting_api/controllers/main.py
Normal file
69
jikimo_work_reporting_api/controllers/main.py
Normal file
@@ -0,0 +1,69 @@
|
|||||||
|
import json
|
||||||
|
from odoo import http
|
||||||
|
from odoo.http import request
|
||||||
|
from odoo.addons.sf_machine_connect.models.ftp_operate import transfer_files
|
||||||
|
from odoo.addons.sf_base.decorators.api_log import api_log
|
||||||
|
|
||||||
|
class MainController(http.Controller):
|
||||||
|
|
||||||
|
@http.route('/api/manual_download_program', type='json', methods=['POST'], auth='wechat_token', cors='*')
|
||||||
|
@api_log('人工线下加工编程文件传输', requester='报工系统')
|
||||||
|
def manual_download_program(self):
|
||||||
|
"""
|
||||||
|
人工线下加工传输编程文件
|
||||||
|
"""
|
||||||
|
data = json.loads(request.httprequest.data)
|
||||||
|
maintenance_equipment_id = data.get('maintenance_equipment_id')
|
||||||
|
model_id = data.get('model_id')
|
||||||
|
if not maintenance_equipment_id or not model_id:
|
||||||
|
return {'code': 400, 'message': '参数错误'}
|
||||||
|
try:
|
||||||
|
model_id = int(model_id)
|
||||||
|
except Exception as e:
|
||||||
|
return {'code': 400, 'message': '参数类型错误'}
|
||||||
|
maintenance_equipment = request.env['maintenance.equipment'].sudo().search(
|
||||||
|
[('MTcode', '=', maintenance_equipment_id), ('category_id.equipment_type', '=', '机床')],
|
||||||
|
limit=1
|
||||||
|
)
|
||||||
|
if not maintenance_equipment:
|
||||||
|
return {'code': 400, 'message': '机台不存在,请扫描正确的机台二维码'}
|
||||||
|
product = request.env['product.template'].sudo().search([('model_id', '=', model_id)], limit=1)
|
||||||
|
if not product:
|
||||||
|
return {'code': 400, 'message': '请扫描正确的图纸'}
|
||||||
|
# 获取刀具组
|
||||||
|
tool_groups_id = request.env['sf.tool.groups'].sudo().search([('equipment_ids', 'in', maintenance_equipment.id)], limit=1)
|
||||||
|
if not tool_groups_id:
|
||||||
|
return {'code': 400, 'message': '刀具组不存在'}
|
||||||
|
|
||||||
|
ftp_resconfig = request.env['res.config.settings'].sudo().get_values()
|
||||||
|
if not ftp_resconfig['ftp_host'] or not ftp_resconfig['ftp_port'] or not ftp_resconfig['ftp_user'] or not ftp_resconfig['ftp_password']:
|
||||||
|
return {'code': 400, 'message': '编程文件FTP配置错误'}
|
||||||
|
source_ftp_info = {
|
||||||
|
'host': ftp_resconfig['ftp_host'],
|
||||||
|
'port': int(ftp_resconfig['ftp_port']),
|
||||||
|
'username': ftp_resconfig['ftp_user'],
|
||||||
|
'password': ftp_resconfig['ftp_password']
|
||||||
|
}
|
||||||
|
if not maintenance_equipment.ftp_host or not maintenance_equipment.ftp_port or not maintenance_equipment.ftp_username or not maintenance_equipment.ftp_password:
|
||||||
|
return {'code': 400, 'message': '机台FTP配置错误'}
|
||||||
|
target_ftp_info = {
|
||||||
|
'host': maintenance_equipment.ftp_host,
|
||||||
|
'port': int(maintenance_equipment.ftp_port),
|
||||||
|
'username': maintenance_equipment.ftp_username,
|
||||||
|
'password': maintenance_equipment.ftp_password
|
||||||
|
}
|
||||||
|
# 传输nc文件
|
||||||
|
try:
|
||||||
|
result = transfer_files(
|
||||||
|
source_ftp_info,
|
||||||
|
target_ftp_info,
|
||||||
|
'/' + str(model_id),
|
||||||
|
'/',
|
||||||
|
match_str=r'^\d*-' + tool_groups_id.name + r'-\w{2}-all\.nc$'
|
||||||
|
)
|
||||||
|
if len(result) > 0:
|
||||||
|
return {'code': 200, 'message': '传输成功', 'file_list': result}
|
||||||
|
else:
|
||||||
|
return {'code': 404, 'message': '未找到编程文件'}
|
||||||
|
except Exception as e:
|
||||||
|
return {'code': 500, 'message': str(e)}
|
||||||
1
jikimo_work_reporting_api/models/__init__.py
Normal file
1
jikimo_work_reporting_api/models/__init__.py
Normal file
@@ -0,0 +1 @@
|
|||||||
|
# -*- coding: utf-8 -*-
|
||||||
@@ -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):
|
||||||
"""
|
"""
|
||||||
记录工单异常
|
记录工单异常
|
||||||
|
|||||||
@@ -8,7 +8,7 @@
|
|||||||
<field name="arch" type="xml">
|
<field name="arch" type="xml">
|
||||||
<xpath expr="//notebook/page[last()]" position="after">
|
<xpath expr="//notebook/page[last()]" position="after">
|
||||||
<field name="routing_type" invisible="1"/>
|
<field name="routing_type" invisible="1"/>
|
||||||
<page string="异常记录" name="workorder_exception" attrs="{'invisible': [('routing_type', '!=', 'CNC加工')]}">
|
<page string="异常记录" name="workorder_exception" attrs='{"invisible": ["!", ("individuation_page_list", "ilike", "ER")]}'>
|
||||||
<field name="exception_ids" nolabel="1" readonly="1">
|
<field name="exception_ids" nolabel="1" readonly="1">
|
||||||
<tree create="false" delete="false" edit="false">
|
<tree create="false" delete="false" edit="false">
|
||||||
<field name="exception_content" string="反馈的异常/问题信息"/>
|
<field name="exception_content" string="反馈的异常/问题信息"/>
|
||||||
|
|||||||
@@ -52,10 +52,10 @@ class JikimoWorkorderException(models.Model):
|
|||||||
|
|
||||||
def _get_message(self, message_queue_ids):
|
def _get_message(self, message_queue_ids):
|
||||||
contents, _ = super(JikimoWorkorderException, self)._get_message(message_queue_ids)
|
contents, _ = super(JikimoWorkorderException, self)._get_message(message_queue_ids)
|
||||||
url = self.env['ir.config_parameter'].get_param('web.base.url')
|
base_url = self.env['ir.config_parameter'].get_param('web.base.url')
|
||||||
action_id = self.env.ref('mrp.mrp_production_action').id
|
action_id = self.env.ref('mrp.mrp_production_action').id
|
||||||
for index, content in enumerate(contents):
|
for index, content in enumerate(contents):
|
||||||
exception_id = self.env['jikimo.workorder.exception'].browse(message_queue_ids[index].res_id)
|
exception_id = self.env['jikimo.workorder.exception'].browse(message_queue_ids[index].res_id)
|
||||||
url = url + '/web#id=%s&view_type=form&action=%s' % (exception_id.workorder_id.production_id.id, action_id)
|
url = base_url + '/web#id=%s&view_type=form&action=%s' % (exception_id.workorder_id.production_id.id, action_id)
|
||||||
contents[index] = content.replace('{{url}}', url)
|
contents[index] = content.replace('{{url}}', url)
|
||||||
return contents, message_queue_ids
|
return contents, message_queue_ids
|
||||||
|
|||||||
@@ -4,3 +4,4 @@
|
|||||||
from . import models
|
from . import models
|
||||||
from . import wizard
|
from . import wizard
|
||||||
from . import report
|
from . import report
|
||||||
|
from . import controllers
|
||||||
|
|||||||
@@ -8,7 +8,7 @@
|
|||||||
'sequence': 120,
|
'sequence': 120,
|
||||||
'summary': 'Control the quality of your products',
|
'summary': 'Control the quality of your products',
|
||||||
'website': 'https://www.odoo.com/app/quality',
|
'website': 'https://www.odoo.com/app/quality',
|
||||||
'depends': ['quality', 'sf_manufacturing'],
|
'depends': ['quality', 'sf_manufacturing', 'base_import'],
|
||||||
'description': """
|
'description': """
|
||||||
Quality Control
|
Quality Control
|
||||||
===============
|
===============
|
||||||
@@ -20,12 +20,15 @@ Quality Control
|
|||||||
""",
|
""",
|
||||||
'data': [
|
'data': [
|
||||||
'data/quality_control_data.xml',
|
'data/quality_control_data.xml',
|
||||||
|
'wizard/import_complex_model.xml',
|
||||||
|
'wizard/quality_wizard_view.xml',
|
||||||
'report/worksheet_custom_reports.xml',
|
'report/worksheet_custom_reports.xml',
|
||||||
'report/worksheet_custom_report_templates.xml',
|
'report/worksheet_custom_report_templates.xml',
|
||||||
'views/quality_views.xml',
|
'views/quality_views.xml',
|
||||||
'views/product_views.xml',
|
'views/product_views.xml',
|
||||||
'views/stock_move_views.xml',
|
'views/stock_move_views.xml',
|
||||||
'views/stock_picking_views.xml',
|
'views/stock_picking_views.xml',
|
||||||
|
'views/quality.check.measures.line.xml',
|
||||||
'wizard/quality_check_wizard_views.xml',
|
'wizard/quality_check_wizard_views.xml',
|
||||||
'security/ir.model.access.csv',
|
'security/ir.model.access.csv',
|
||||||
],
|
],
|
||||||
|
|||||||
1
quality_control/controllers/__init__.py
Normal file
1
quality_control/controllers/__init__.py
Normal file
@@ -0,0 +1 @@
|
|||||||
|
from . import main
|
||||||
120
quality_control/controllers/main.py
Normal file
120
quality_control/controllers/main.py
Normal file
@@ -0,0 +1,120 @@
|
|||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
|
||||||
|
from odoo import http
|
||||||
|
from odoo.http import request, Response
|
||||||
|
import base64
|
||||||
|
import json
|
||||||
|
|
||||||
|
|
||||||
|
class QualityController(http.Controller):
|
||||||
|
|
||||||
|
@http.route(['/api/quality/report/download'], type='http', auth='public', csrf=False, website=False) # 移除 cors="*"
|
||||||
|
def get_quality_report(self, retrospect_ref=None, **kwargs):
|
||||||
|
"""获取质检报告的下载接口
|
||||||
|
|
||||||
|
Args:
|
||||||
|
retrospect_ref: 追溯码
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
直接返回文件下载响应
|
||||||
|
"""
|
||||||
|
try:
|
||||||
|
# 如果retrospect_ref为None,尝试从查询参数获取
|
||||||
|
if not retrospect_ref:
|
||||||
|
retrospect_ref = kwargs.get('retrospect_ref')
|
||||||
|
|
||||||
|
# 参数验证
|
||||||
|
if not retrospect_ref:
|
||||||
|
return self._json_response({
|
||||||
|
'status': 'error',
|
||||||
|
'message': '追溯码不能为空'
|
||||||
|
})
|
||||||
|
|
||||||
|
# 查找对应的质检单
|
||||||
|
quality_check = request.env['quality.check'].sudo().search([
|
||||||
|
('picking_id.retrospect_ref', '=', retrospect_ref),
|
||||||
|
('publish_status', '=', 'published') # 只返回已发布的报告
|
||||||
|
], limit=1)
|
||||||
|
|
||||||
|
if not quality_check:
|
||||||
|
return self._json_response({
|
||||||
|
'status': 'error',
|
||||||
|
'message': '未找到对应的质检报告或报告未发布'
|
||||||
|
})
|
||||||
|
|
||||||
|
if not quality_check.report_number_id:
|
||||||
|
return self._json_response({
|
||||||
|
'status': 'error',
|
||||||
|
'message': '质检报告文件不存在'
|
||||||
|
})
|
||||||
|
|
||||||
|
# 获取文件内容
|
||||||
|
document = quality_check.report_number_id
|
||||||
|
if not document.raw: # 检查文件内容是否存在
|
||||||
|
return self._json_response({
|
||||||
|
'status': 'error',
|
||||||
|
'message': '文件内容不存在'
|
||||||
|
})
|
||||||
|
|
||||||
|
# 构建文件名(确保有.pdf后缀)
|
||||||
|
filename = document.name
|
||||||
|
if not filename.lower().endswith('.pdf'):
|
||||||
|
filename = f"{filename}.pdf"
|
||||||
|
|
||||||
|
# 返回文件下载响应
|
||||||
|
return Response(
|
||||||
|
document.raw,
|
||||||
|
headers=[
|
||||||
|
('Content-Type', 'application/pdf'),
|
||||||
|
('Content-Disposition', f'attachment; filename="{filename}"'),
|
||||||
|
('Access-Control-Allow-Origin', '*'),
|
||||||
|
('Access-Control-Allow-Methods', 'GET, OPTIONS'),
|
||||||
|
('Access-Control-Allow-Headers', 'Content-Type, Authorization')
|
||||||
|
]
|
||||||
|
)
|
||||||
|
|
||||||
|
except Exception as e:
|
||||||
|
return self._json_response({
|
||||||
|
'status': 'error',
|
||||||
|
'message': f'系统错误: {str(e)}'
|
||||||
|
})
|
||||||
|
|
||||||
|
def _json_response(self, data):
|
||||||
|
"""返回JSON格式的响应"""
|
||||||
|
return Response(
|
||||||
|
json.dumps(data, ensure_ascii=False),
|
||||||
|
mimetype='application/json;charset=utf-8',
|
||||||
|
headers=[
|
||||||
|
('Access-Control-Allow-Origin', '*'),
|
||||||
|
('Access-Control-Allow-Methods', 'GET, OPTIONS'),
|
||||||
|
('Access-Control-Allow-Headers', 'Content-Type, Authorization')
|
||||||
|
]
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
class QualityReportController(http.Controller):
|
||||||
|
@http.route('/quality/report/<int:document_id>', type='http', auth='public')
|
||||||
|
def get_public_report(self, document_id, **kw):
|
||||||
|
"""提供公开访问PDF报告的控制器"""
|
||||||
|
document = request.env['documents.document'].sudo().browse(int(document_id))
|
||||||
|
|
||||||
|
# 安全检查:确保只有质检报告文档可以被访问
|
||||||
|
if document.exists() and document.res_model == 'quality.check':
|
||||||
|
# 获取PDF内容
|
||||||
|
pdf_content = document.raw
|
||||||
|
|
||||||
|
# 返回PDF内容
|
||||||
|
return request.make_response(
|
||||||
|
pdf_content,
|
||||||
|
headers=[
|
||||||
|
('Content-Type', 'application/pdf'),
|
||||||
|
('Content-Disposition', f'inline; filename={document.name}.pdf')
|
||||||
|
]
|
||||||
|
)
|
||||||
|
return request.not_found()
|
||||||
|
|
||||||
|
@http.route('/quality/report/not_published', type='http', auth='public')
|
||||||
|
def get_not_published_report(self, **kw):
|
||||||
|
"""提供未发布报告的控制器"""
|
||||||
|
return "报告尚未发布"
|
||||||
|
|
||||||
@@ -10,6 +10,12 @@ from odoo import api, models, fields, _
|
|||||||
from odoo.api import depends
|
from odoo.api import depends
|
||||||
from odoo.tools import DEFAULT_SERVER_DATETIME_FORMAT, float_round
|
from odoo.tools import DEFAULT_SERVER_DATETIME_FORMAT, float_round
|
||||||
from odoo.osv.expression import OR
|
from odoo.osv.expression import OR
|
||||||
|
from odoo.exceptions import UserError
|
||||||
|
from odoo.tools import image_data_uri
|
||||||
|
from base64 import b64encode
|
||||||
|
import requests
|
||||||
|
import json
|
||||||
|
import base64
|
||||||
|
|
||||||
|
|
||||||
class QualityPoint(models.Model):
|
class QualityPoint(models.Model):
|
||||||
@@ -34,7 +40,8 @@ class QualityPoint(models.Model):
|
|||||||
('day', 'Days'),
|
('day', 'Days'),
|
||||||
('week', 'Weeks'),
|
('week', 'Weeks'),
|
||||||
('month', 'Months')], default="day") # TDE RENAME ?
|
('month', 'Months')], default="day") # TDE RENAME ?
|
||||||
is_lot_tested_fractionally = fields.Boolean(string="Lot Tested Fractionally", help="Determines if only a fraction of the lot should be tested")
|
is_lot_tested_fractionally = fields.Boolean(string="Lot Tested Fractionally",
|
||||||
|
help="Determines if only a fraction of the lot should be tested")
|
||||||
testing_percentage_within_lot = fields.Float(help="Defines the percentage within a lot that should be tested")
|
testing_percentage_within_lot = fields.Float(help="Defines the percentage within a lot that should be tested")
|
||||||
norm = fields.Float('Norm', digits='Quality Tests') # TDE RENAME ?
|
norm = fields.Float('Norm', digits='Quality Tests') # TDE RENAME ?
|
||||||
tolerance_min = fields.Float('Min Tolerance', digits='Quality Tests')
|
tolerance_min = fields.Float('Min Tolerance', digits='Quality Tests')
|
||||||
@@ -63,7 +70,7 @@ class QualityPoint(models.Model):
|
|||||||
|
|
||||||
if n > 1:
|
if n > 1:
|
||||||
point.average = mean
|
point.average = mean
|
||||||
point.standard_deviation = sqrt( s / ( n - 1))
|
point.standard_deviation = sqrt(s / (n - 1))
|
||||||
elif n == 1:
|
elif n == 1:
|
||||||
point.average = mean
|
point.average = mean
|
||||||
point.standard_deviation = 0.0
|
point.standard_deviation = 0.0
|
||||||
@@ -94,7 +101,7 @@ class QualityPoint(models.Model):
|
|||||||
checks = self.env['quality.check'].search([
|
checks = self.env['quality.check'].search([
|
||||||
('point_id', '=', self.id),
|
('point_id', '=', self.id),
|
||||||
('create_date', '>=', date_previous.strftime(DEFAULT_SERVER_DATETIME_FORMAT))], limit=1)
|
('create_date', '>=', date_previous.strftime(DEFAULT_SERVER_DATETIME_FORMAT))], limit=1)
|
||||||
return not(bool(checks))
|
return not (bool(checks))
|
||||||
return super(QualityPoint, self).check_execute_now()
|
return super(QualityPoint, self).check_execute_now()
|
||||||
|
|
||||||
def _get_type_default_domain(self):
|
def _get_type_default_domain(self):
|
||||||
@@ -123,13 +130,509 @@ class QualityPoint(models.Model):
|
|||||||
|
|
||||||
class QualityCheck(models.Model):
|
class QualityCheck(models.Model):
|
||||||
_inherit = "quality.check"
|
_inherit = "quality.check"
|
||||||
part_name = fields.Char('零件名称', compute='_compute_part_name_number', readonly=True)
|
part_name = fields.Char('零件名称', related='product_id.part_name', readonly=False, store=True)
|
||||||
part_number = fields.Char('零件图号', compute='_compute_part_name_number', readonly=True)
|
part_number = fields.Char('零件图号', related='product_id.part_number', readonly=False, store=True)
|
||||||
@depends('product_id')
|
material_name = fields.Char('材料名称', compute='_compute_material_name')
|
||||||
def _compute_part_name_number(self):
|
model_id = fields.Char('模型ID', related='product_id.model_id')
|
||||||
|
|
||||||
|
# # 总数量,值为调拨单_产品明细_数量
|
||||||
|
# total_qty = fields.Float('总数量', compute='_compute_total_qty', readonly=True)
|
||||||
|
# # 检验数
|
||||||
|
# check_qty = fields.Float('检验数', compute='_compute_check_qty', readonly=True)
|
||||||
|
# # 出厂检验报告编号
|
||||||
|
# report_number = fields.Char('出厂检验报告编号', compute='_compute_report_number', readonly=True)
|
||||||
|
# 总数量,值为调拨单_产品明细_数量
|
||||||
|
total_qty = fields.Char('总数量', compute='_compute_total_qty', store=True)
|
||||||
|
|
||||||
|
column_nums = fields.Integer('测量值列数', default=1)
|
||||||
|
|
||||||
|
@api.depends('picking_id')
|
||||||
|
def _compute_total_qty(self):
|
||||||
for record in self:
|
for record in self:
|
||||||
record.part_number = record.product_id.part_number
|
if record.picking_id:
|
||||||
record.part_name = record.product_id.part_name
|
total_qty = 0
|
||||||
|
for move in record.picking_id.move_ids_without_package:
|
||||||
|
if move.product_id == record.product_id:
|
||||||
|
total_qty = int(move.product_uom_qty)
|
||||||
|
record.total_qty = total_qty if total_qty > 0 else 0
|
||||||
|
else:
|
||||||
|
record.total_qty = 0
|
||||||
|
|
||||||
|
# 检验数
|
||||||
|
check_qty = fields.Integer('检验数', default=lambda self: self._get_default_check_qty())
|
||||||
|
|
||||||
|
def _get_default_check_qty(self):
|
||||||
|
"""根据条件设置检验数的默认值"""
|
||||||
|
# 这里需要使用_origin来获取已存储的记录,因为新记录在创建时可能还没有这些值
|
||||||
|
if self._origin:
|
||||||
|
if self._origin.measure_on == 'product' and self._origin.test_type_id.name == '出厂检验报告':
|
||||||
|
return ''
|
||||||
|
elif self._origin.measure_on == 'product':
|
||||||
|
return '1'
|
||||||
|
return ''
|
||||||
|
|
||||||
|
@api.onchange('test_type_id', 'measure_on')
|
||||||
|
def _onchange_check_qty(self):
|
||||||
|
"""当测试类型或测量对象变化时,更新检验数"""
|
||||||
|
if self.measure_on == 'product' and self.test_type_id.name == '出厂检验报告':
|
||||||
|
self.check_qty = 0
|
||||||
|
elif self.measure_on == 'product':
|
||||||
|
self.check_qty = 1
|
||||||
|
|
||||||
|
# 出厂检验报告编号
|
||||||
|
report_number_id = fields.Many2one('documents.document', string='出厂检验报告编号', readonly=True)
|
||||||
|
report_number_name = fields.Char('出厂检验报告编号名称', compute='_compute_report_number_name')
|
||||||
|
|
||||||
|
old_report_name = fields.Char('旧出厂检验报告编号', default='')
|
||||||
|
|
||||||
|
@api.depends('serial_number', 'part_number')
|
||||||
|
def _compute_report_number_name(self):
|
||||||
|
for record in self:
|
||||||
|
str_serial_number = '0' + str(record.serial_number) if record.serial_number < 10 else str(
|
||||||
|
record.serial_number)
|
||||||
|
str_part_number = record.part_number if record.part_number else ''
|
||||||
|
record.report_number_name = f'FQC{str_part_number}{str_serial_number}'
|
||||||
|
|
||||||
|
# 出厂检验报告、关联文档的数据
|
||||||
|
report_content = fields.Binary(string='出厂检验报告', related='report_number_id.datas')
|
||||||
|
|
||||||
|
is_out_check = fields.Boolean(string='是否出库检验', compute='_compute_is_out_check', readonly=True)
|
||||||
|
|
||||||
|
measure_line_ids = fields.One2many('quality.check.measure.line', 'check_id', string='测量明细')
|
||||||
|
|
||||||
|
categ_type = fields.Selection(string='产品的类别', related='product_id.categ_id.type', store=True)
|
||||||
|
|
||||||
|
report_result = fields.Selection([
|
||||||
|
('OK', 'OK'),
|
||||||
|
('NG', 'NG')
|
||||||
|
], string='出厂检验报告结果', default='OK')
|
||||||
|
measure_operator = fields.Many2one('res.users', string='操机员')
|
||||||
|
quality_manager = fields.Many2one('res.users', string='质检员', compute='_compute_quality_manager')
|
||||||
|
|
||||||
|
@api.depends('measure_line_ids')
|
||||||
|
def _compute_quality_manager(self):
|
||||||
|
for record in self:
|
||||||
|
if record.measure_line_ids:
|
||||||
|
record.quality_manager = record.env.user.id
|
||||||
|
else:
|
||||||
|
record.quality_manager = False
|
||||||
|
|
||||||
|
# 流水号(从1开始,最大99)
|
||||||
|
serial_number = fields.Integer('流水号', default=1, readonly=True)
|
||||||
|
|
||||||
|
# 发布历史
|
||||||
|
report_history_ids = fields.One2many('quality.check.report.history', 'check_id', string='发布历史')
|
||||||
|
|
||||||
|
# 发布状态
|
||||||
|
publish_status = fields.Selection([
|
||||||
|
('draft', '草稿'),
|
||||||
|
('published', '已发布'),
|
||||||
|
('canceled', '已撤销')
|
||||||
|
], string='发布状态', default='draft')
|
||||||
|
|
||||||
|
# 出厂检验报告是否已上传
|
||||||
|
is_factory_report_uploaded = fields.Boolean(string='出厂检验报告是否已上传', default=False)
|
||||||
|
|
||||||
|
def add_measure_line(self):
|
||||||
|
"""
|
||||||
|
新增测量值,如果测量值有5列了,则提示“最多只能有5列测量值”
|
||||||
|
"""
|
||||||
|
if self.column_nums >= 5:
|
||||||
|
raise UserError(_('最多只能有5列测量值'))
|
||||||
|
else:
|
||||||
|
for line in self.measure_line_ids:
|
||||||
|
field_name = f'measure_value{self.column_nums + 1}'
|
||||||
|
if hasattr(line, field_name):
|
||||||
|
line[field_name] = False
|
||||||
|
self.column_nums = self.column_nums + 1
|
||||||
|
|
||||||
|
def remove_measure_line(self):
|
||||||
|
"""
|
||||||
|
删除测量值
|
||||||
|
"""
|
||||||
|
if self.column_nums <= 1:
|
||||||
|
raise UserError(_('最少要有1列测量值'))
|
||||||
|
else:
|
||||||
|
for line in self.measure_line_ids:
|
||||||
|
field_name = f'measure_value{self.column_nums}'
|
||||||
|
if hasattr(line, field_name):
|
||||||
|
line[field_name] = False
|
||||||
|
self.column_nums = self.column_nums - 1
|
||||||
|
|
||||||
|
def upload_measure_line(self):
|
||||||
|
"""
|
||||||
|
上传测量值
|
||||||
|
"""
|
||||||
|
|
||||||
|
for record in self:
|
||||||
|
if not record.part_name or not record.part_number:
|
||||||
|
raise UserError(_('零件名称和零件图号均不能为空'))
|
||||||
|
|
||||||
|
# 如果验证通过,返回原动作
|
||||||
|
action = self.env.ref('quality_control.import_complex_model_wizard').read()[0]
|
||||||
|
action['context'] = {
|
||||||
|
'default_model_name': 'quality.check.measure.line',
|
||||||
|
'default_check_id': self.id,
|
||||||
|
}
|
||||||
|
return action
|
||||||
|
|
||||||
|
def do_preview(self):
|
||||||
|
"""
|
||||||
|
预览出厂检验报告
|
||||||
|
"""
|
||||||
|
pass
|
||||||
|
|
||||||
|
def do_publish(self):
|
||||||
|
"""发布出厂检验报告"""
|
||||||
|
self.ensure_one()
|
||||||
|
self._check_part_number()
|
||||||
|
self._check_measure_line()
|
||||||
|
self._check_check_qty_and_total_qty()
|
||||||
|
|
||||||
|
# 打开确认向导而不是直接发布
|
||||||
|
return {
|
||||||
|
'name': _('发布确认'),
|
||||||
|
'type': 'ir.actions.act_window',
|
||||||
|
'res_model': 'quality.check.publish.wizard',
|
||||||
|
'view_mode': 'form',
|
||||||
|
'target': 'new',
|
||||||
|
'context': {
|
||||||
|
'default_check_id': self.id,
|
||||||
|
'default_product_name': self.product_id.name,
|
||||||
|
'default_total_qty': self.total_qty,
|
||||||
|
'default_check_qty': self.check_qty,
|
||||||
|
'default_measure_count': self.column_nums,
|
||||||
|
'default_item_count': len(self.measure_line_ids),
|
||||||
|
'default_old_report_name': self.old_report_name,
|
||||||
|
'default_publish_status': self.publish_status,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
def _do_publish_implementation(self):
|
||||||
|
"""实际执行发布操作的方法"""
|
||||||
|
self.ensure_one()
|
||||||
|
|
||||||
|
# 1. 获取已发布的文档文件夹
|
||||||
|
workspace = self.env['documents.folder'].search(
|
||||||
|
[('parent_folder_id', '=', self.env.ref('sf_quality.documents_purchase_contracts_folder').id),
|
||||||
|
('name', '=', '已发布')], limit=1)
|
||||||
|
|
||||||
|
if self.serial_number > 99:
|
||||||
|
raise UserError(_('流水号不能大于99'))
|
||||||
|
|
||||||
|
# 2. 先创建空文档记录
|
||||||
|
doc_vals = {
|
||||||
|
'name': self.report_number_name,
|
||||||
|
'mimetype': 'application/pdf',
|
||||||
|
'res_id': self.id,
|
||||||
|
'folder_id': workspace.id,
|
||||||
|
'res_model': self._name,
|
||||||
|
}
|
||||||
|
|
||||||
|
doc = self.env['documents.document'].create(doc_vals)
|
||||||
|
|
||||||
|
# 3. 关联文档到质检记录
|
||||||
|
self.write({
|
||||||
|
'report_number_id': doc.id,
|
||||||
|
'quality_state': 'pass'
|
||||||
|
})
|
||||||
|
|
||||||
|
# 4. 获取报告动作并生成PDF(此时二维码将包含正确的文档ID)
|
||||||
|
report_action = self.env.ref('sf_quality.action_report_quality_inspection')
|
||||||
|
pdf_content, v = report_action._render_qweb_pdf(
|
||||||
|
report_ref=report_action.report_name,
|
||||||
|
res_ids=self.ids
|
||||||
|
)
|
||||||
|
|
||||||
|
# 5. 更新文档内容
|
||||||
|
doc.write({
|
||||||
|
'raw': pdf_content
|
||||||
|
})
|
||||||
|
|
||||||
|
# 6. 记录发布历史
|
||||||
|
self.env['quality.check.report.history'].create({
|
||||||
|
'check_id': self.id,
|
||||||
|
'report_number_id': doc.id,
|
||||||
|
'action': 'publish',
|
||||||
|
'operator': self.env.user.name,
|
||||||
|
'operation_time': datetime.now(),
|
||||||
|
'document_status': 'published',
|
||||||
|
'sequence': len(self.report_history_ids) + 1
|
||||||
|
})
|
||||||
|
|
||||||
|
# 7. 更新其他信息
|
||||||
|
self.serial_number += 1
|
||||||
|
|
||||||
|
if self.publish_status == 'canceled' and self.picking_id.state == 'done':
|
||||||
|
self.upload_factory_report()
|
||||||
|
|
||||||
|
self.write({
|
||||||
|
'publish_status': 'published',
|
||||||
|
})
|
||||||
|
|
||||||
|
return True
|
||||||
|
|
||||||
|
# 发布前检验零件图号、操机员、质检员
|
||||||
|
def _check_part_number(self):
|
||||||
|
if not self.part_number:
|
||||||
|
raise UserError(_('零件图号不能为空'))
|
||||||
|
if not self.measure_operator:
|
||||||
|
raise UserError(_('操机员不能为空'))
|
||||||
|
|
||||||
|
# 发布前校验明细行列均非空
|
||||||
|
def _check_measure_line(self):
|
||||||
|
for record in self:
|
||||||
|
if not record.measure_line_ids:
|
||||||
|
raise UserError(_('请先添加测量明细'))
|
||||||
|
for line in record.measure_line_ids:
|
||||||
|
if not line.measure_item:
|
||||||
|
raise UserError(_('有检测项目值为空'))
|
||||||
|
for i in range(1, record.column_nums + 1):
|
||||||
|
if not getattr(line, f'measure_value{i}'):
|
||||||
|
raise UserError(_('有测量值为空'))
|
||||||
|
|
||||||
|
# 发布前校验检验数与总数量、检验数与测量件数(即测量列数)
|
||||||
|
def _check_check_qty_and_total_qty(self):
|
||||||
|
for record in self:
|
||||||
|
if not record.check_qty:
|
||||||
|
raise UserError(_('请先输入检验数'))
|
||||||
|
if not record.total_qty:
|
||||||
|
raise UserError(_('总数量不能为空'))
|
||||||
|
if record.check_qty > int(record.total_qty):
|
||||||
|
raise UserError(_('检验数不可超过总数量'))
|
||||||
|
if record.column_nums > record.check_qty:
|
||||||
|
raise UserError(_('测量件数不可超过检验数'))
|
||||||
|
|
||||||
|
def do_cancel_publish(self):
|
||||||
|
"""
|
||||||
|
取消发布出厂检验报告(将当前质检单关联的出厂检验报告文档位置移动到废弃文件夹), 并记录发布历史
|
||||||
|
"""
|
||||||
|
self.ensure_one()
|
||||||
|
# 1. 获取已发布的文档文件夹
|
||||||
|
workspace = self.env['documents.folder'].search(
|
||||||
|
[('parent_folder_id', '=', self.env.ref('sf_quality.documents_purchase_contracts_folder').id),
|
||||||
|
('name', '=', '已发布')], limit=1)
|
||||||
|
# 2. 将当前质检单关联的出厂检验报告文档位置移动到废弃文件夹
|
||||||
|
self.report_number_id.write({
|
||||||
|
'folder_id': self.env.ref('sf_quality.documents_purchase_contracts_folder_canceled').id,
|
||||||
|
})
|
||||||
|
|
||||||
|
# 3. 记录发布历史
|
||||||
|
self.env['quality.check.report.history'].create({
|
||||||
|
'check_id': self.id,
|
||||||
|
'report_number_id': self.report_number_id.id,
|
||||||
|
'action': 'cancel_publish',
|
||||||
|
'operator': self.env.user.name,
|
||||||
|
'operation_time': datetime.now(),
|
||||||
|
'document_status': 'canceled',
|
||||||
|
'sequence': len(self.report_history_ids) + 1
|
||||||
|
})
|
||||||
|
|
||||||
|
self.write({
|
||||||
|
'old_report_name': self.report_number_id.name
|
||||||
|
})
|
||||||
|
|
||||||
|
# 3. 更新发布状态
|
||||||
|
self.write({
|
||||||
|
'publish_status': 'canceled',
|
||||||
|
'report_number_id': False,
|
||||||
|
'quality_state': 'none'
|
||||||
|
})
|
||||||
|
|
||||||
|
if self.is_factory_report_uploaded:
|
||||||
|
# 4. 删除加工订单明细中的出厂检验报告
|
||||||
|
self.delete_factory_report()
|
||||||
|
|
||||||
|
return True
|
||||||
|
|
||||||
|
def do_re_publish(self):
|
||||||
|
"""
|
||||||
|
重新发布出厂检验报告,参考发布规则
|
||||||
|
"""
|
||||||
|
return self.do_publish()
|
||||||
|
|
||||||
|
def generate_qr_code(self):
|
||||||
|
"""生成二维码URL"""
|
||||||
|
self.ensure_one()
|
||||||
|
base_url = self.env['ir.config_parameter'].sudo().get_param('web.base.url')
|
||||||
|
return image_data_uri(
|
||||||
|
b64encode(self.env['ir.actions.report'].barcode(
|
||||||
|
'QR', base_url + '/#/index/publicPay?order_id=' + str(self.id) + '&source=%2Findex%2Fmyorder',
|
||||||
|
width=140, height=140)
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
|
def get_latest_report_attachment(self, check_id):
|
||||||
|
"""获取指定质检记录的最新报告附件,并删除旧的报告附件"""
|
||||||
|
# 查找特定质检记录的所有附件
|
||||||
|
attachments = self.env['ir.attachment'].search([
|
||||||
|
('res_model', '=', 'quality.check'),
|
||||||
|
('res_id', '=', check_id),
|
||||||
|
('name', 'like', 'QC-QC') # 根据您的命名规则调整
|
||||||
|
], order='create_date DESC') # 按创建日期降序排序
|
||||||
|
|
||||||
|
# # 如果附件数量大于1,则删除除最新报告外的其他报告附件
|
||||||
|
# if len(attachments) > 1:
|
||||||
|
# for attachment in attachments[1:]:
|
||||||
|
# attachment.unlink()
|
||||||
|
|
||||||
|
# 返回最新的附件(如果存在)
|
||||||
|
return attachments and attachments[0] or False
|
||||||
|
|
||||||
|
def get_report_url(self):
|
||||||
|
"""生成报告访问URL"""
|
||||||
|
self.ensure_one()
|
||||||
|
base_url = self.env['ir.config_parameter'].sudo().get_param('web.base.url')
|
||||||
|
if self.report_number_id:
|
||||||
|
print(f"{base_url}/quality/report/{self.report_number_id.id}")
|
||||||
|
return f"{base_url}/quality/report/{self.report_number_id.id}"
|
||||||
|
else:
|
||||||
|
return f"{base_url}/quality/report/not_published"
|
||||||
|
|
||||||
|
def upload_factory_report(self):
|
||||||
|
"""
|
||||||
|
上传出厂检验报告到加工订单明细中
|
||||||
|
将当前质检单的出厂检验报告上传到对应的加工订单明细中
|
||||||
|
"""
|
||||||
|
self.ensure_one()
|
||||||
|
if not self.report_content:
|
||||||
|
raise UserError(_('当前质检单没有出厂检验报告,请先发布报告'))
|
||||||
|
|
||||||
|
if not self.product_id.model_name:
|
||||||
|
raise UserError(_('产品模型名称为空'))
|
||||||
|
|
||||||
|
if not self.picking_id or not self.picking_id.origin:
|
||||||
|
raise UserError(_('无法找到相关的调拨单或来源单据'))
|
||||||
|
|
||||||
|
# 获取订单号(从调拨单的来源字段获取)
|
||||||
|
order_ref = self.picking_id.retrospect_ref
|
||||||
|
|
||||||
|
try:
|
||||||
|
# 准备请求数据
|
||||||
|
payload = {
|
||||||
|
"order_ref": order_ref,
|
||||||
|
"model_name": self.product_id.model_name,
|
||||||
|
"report_file": self.report_content.decode('utf-8') if isinstance(self.report_content,
|
||||||
|
bytes) else self.report_content
|
||||||
|
}
|
||||||
|
|
||||||
|
# 将Python字典转换为JSON字符串
|
||||||
|
json_data = json.dumps(payload)
|
||||||
|
|
||||||
|
# 获取服务器URL
|
||||||
|
base_url = self.env['ir.config_parameter'].sudo().get_param('bfm_url_new')
|
||||||
|
api_url = f"{base_url}/api/report/create"
|
||||||
|
|
||||||
|
# 设置请求头
|
||||||
|
headers = {
|
||||||
|
'Content-Type': 'application/json',
|
||||||
|
}
|
||||||
|
|
||||||
|
# 发送POST请求
|
||||||
|
response = requests.post(api_url, data=json_data, headers=headers)
|
||||||
|
|
||||||
|
# 处理响应
|
||||||
|
if response.status_code == 200:
|
||||||
|
result = response.json()
|
||||||
|
if result.get('success'):
|
||||||
|
# 上传成功,显示成功消息
|
||||||
|
self.is_factory_report_uploaded = True
|
||||||
|
return {
|
||||||
|
'type': 'ir.actions.client',
|
||||||
|
'tag': 'display_notification',
|
||||||
|
'params': {
|
||||||
|
'title': _('上传成功'),
|
||||||
|
'message': _('出厂检验报告已成功上传到加工订单明细'),
|
||||||
|
'type': 'success',
|
||||||
|
'sticky': False,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else:
|
||||||
|
# API返回失败信息
|
||||||
|
raise UserError(_('上传失败: %s') % result.get('message', '未知错误'))
|
||||||
|
else:
|
||||||
|
# HTTP请求失败
|
||||||
|
raise UserError(_('请求失败,状态码: %s') % response.status_code)
|
||||||
|
|
||||||
|
except Exception as e:
|
||||||
|
raise UserError(_('上传过程中发生错误: %s') % str(e))
|
||||||
|
|
||||||
|
def delete_factory_report(self):
|
||||||
|
"""
|
||||||
|
删除加工订单明细中的出厂检验报告
|
||||||
|
"""
|
||||||
|
# 获取订单号(从调拨单的来源字段获取)
|
||||||
|
order_ref = self.picking_id.retrospect_ref
|
||||||
|
|
||||||
|
if not order_ref:
|
||||||
|
raise UserError(_('无法找到相关的调拨单或来源单据'))
|
||||||
|
|
||||||
|
if not self.product_id.model_name:
|
||||||
|
raise UserError(_('产品模型名称为空'))
|
||||||
|
|
||||||
|
try:
|
||||||
|
# 准备请求数据
|
||||||
|
payload = {
|
||||||
|
"order_ref": order_ref,
|
||||||
|
"model_name": self.product_id.model_name
|
||||||
|
}
|
||||||
|
|
||||||
|
# 将Python字典转换为JSON字符串
|
||||||
|
json_data = json.dumps(payload)
|
||||||
|
|
||||||
|
# 获取服务器URL
|
||||||
|
base_url = self.env['ir.config_parameter'].sudo().get_param('bfm_url_new')
|
||||||
|
api_url = f"{base_url}/api/report/delete"
|
||||||
|
|
||||||
|
# 设置请求头
|
||||||
|
headers = {
|
||||||
|
'Content-Type': 'application/json',
|
||||||
|
}
|
||||||
|
|
||||||
|
# 发送POST请求
|
||||||
|
response = requests.post(api_url, data=json_data, headers=headers)
|
||||||
|
|
||||||
|
# 处理响应
|
||||||
|
if response.status_code == 200:
|
||||||
|
result = response.json()
|
||||||
|
if result.get('success'):
|
||||||
|
# 删除成功,显示成功消息
|
||||||
|
self.is_factory_report_uploaded = False
|
||||||
|
return {
|
||||||
|
'type': 'ir.actions.client',
|
||||||
|
'tag': 'display_notification',
|
||||||
|
'params': {
|
||||||
|
'title': _('删除成功'),
|
||||||
|
'message': _('出厂检验报告已成功删除'),
|
||||||
|
'type': 'success',
|
||||||
|
'sticky': False,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else:
|
||||||
|
# API返回失败信息
|
||||||
|
raise UserError(_('删除失败: %s') % result.get('message', '未知错误'))
|
||||||
|
else:
|
||||||
|
# HTTP请求失败
|
||||||
|
raise UserError(_('请求失败,状态码: %s') % response.status_code)
|
||||||
|
|
||||||
|
except Exception as e:
|
||||||
|
raise UserError(_('删除过程中发生错误: %s') % str(e))
|
||||||
|
|
||||||
|
@depends('product_id')
|
||||||
|
def _compute_material_name(self):
|
||||||
|
for record in self:
|
||||||
|
materials_id_name = record.product_id.materials_id.name if record.product_id.materials_id else ''
|
||||||
|
materials_type_name = record.product_id.materials_type_id.name if record.product_id.materials_type_id else ''
|
||||||
|
record.material_name = materials_id_name + ' ' + materials_type_name
|
||||||
|
|
||||||
|
@depends('test_type_id')
|
||||||
|
def _compute_is_out_check(self):
|
||||||
|
for record in self:
|
||||||
|
if record.test_type_id.name == '出厂检验报告':
|
||||||
|
record.is_out_check = True
|
||||||
|
else:
|
||||||
|
record.is_out_check = False
|
||||||
|
|
||||||
failure_message = fields.Html(related='point_id.failure_message', readonly=True)
|
failure_message = fields.Html(related='point_id.failure_message', readonly=True)
|
||||||
measure = fields.Float('Measure', default=0.0, digits='Quality Tests', tracking=True)
|
measure = fields.Float('Measure', default=0.0, digits='Quality Tests', tracking=True)
|
||||||
measure_success = fields.Selection([
|
measure_success = fields.Selection([
|
||||||
@@ -141,7 +644,8 @@ class QualityCheck(models.Model):
|
|||||||
tolerance_max = fields.Float('Max Tolerance', related='point_id.tolerance_max', readonly=True)
|
tolerance_max = fields.Float('Max Tolerance', related='point_id.tolerance_max', readonly=True)
|
||||||
warning_message = fields.Text(compute='_compute_warning_message')
|
warning_message = fields.Text(compute='_compute_warning_message')
|
||||||
norm_unit = fields.Char(related='point_id.norm_unit', readonly=True)
|
norm_unit = fields.Char(related='point_id.norm_unit', readonly=True)
|
||||||
qty_to_test = fields.Float(compute="_compute_qty_to_test", string="Quantity to Test", help="Quantity of product to test within the lot")
|
qty_to_test = fields.Float(compute="_compute_qty_to_test", string="Quantity to Test",
|
||||||
|
help="Quantity of product to test within the lot")
|
||||||
qty_tested = fields.Float(string="Quantity Tested", help="Quantity of product tested within the lot")
|
qty_tested = fields.Float(string="Quantity Tested", help="Quantity of product tested within the lot")
|
||||||
measure_on = fields.Selection([
|
measure_on = fields.Selection([
|
||||||
('operation', 'Operation'),
|
('operation', 'Operation'),
|
||||||
@@ -150,7 +654,8 @@ class QualityCheck(models.Model):
|
|||||||
help="""Operation = One quality check is requested at the operation level.
|
help="""Operation = One quality check is requested at the operation level.
|
||||||
Product = A quality check is requested per product.
|
Product = A quality check is requested per product.
|
||||||
Quantity = A quality check is requested for each new product quantity registered, with partial quantity checks also possible.""")
|
Quantity = A quality check is requested for each new product quantity registered, with partial quantity checks also possible.""")
|
||||||
move_line_id = fields.Many2one('stock.move.line', 'Stock Move Line', check_company=True, help="In case of Quality Check by Quantity, Move Line on which the Quality Check applies")
|
move_line_id = fields.Many2one('stock.move.line', 'Stock Move Line', check_company=True,
|
||||||
|
help="In case of Quality Check by Quantity, Move Line on which the Quality Check applies")
|
||||||
lot_name = fields.Char('Lot/Serial Number Name')
|
lot_name = fields.Char('Lot/Serial Number Name')
|
||||||
lot_line_id = fields.Many2one('stock.lot', store=True, compute='_compute_lot_line_id')
|
lot_line_id = fields.Many2one('stock.lot', store=True, compute='_compute_lot_line_id')
|
||||||
qty_line = fields.Float(compute='_compute_qty_line', string="Quantity")
|
qty_line = fields.Float(compute='_compute_qty_line', string="Quantity")
|
||||||
@@ -231,7 +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:
|
||||||
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")
|
rounding = qc.product_id.uom_id.rounding if qc.product_id.uom_id else 0.01
|
||||||
|
qc.qty_to_test = float_round(qc.qty_line * qc.testing_percentage_within_lot / 100,
|
||||||
|
precision_rounding=rounding, rounding_method="UP")
|
||||||
else:
|
else:
|
||||||
qc.qty_to_test = qc.qty_line
|
qc.qty_to_test = qc.qty_line
|
||||||
|
|
||||||
@@ -386,7 +893,8 @@ class QualityAlert(models.Model):
|
|||||||
class ProductTemplate(models.Model):
|
class ProductTemplate(models.Model):
|
||||||
_inherit = "product.template"
|
_inherit = "product.template"
|
||||||
|
|
||||||
quality_control_point_qty = fields.Integer(compute='_compute_quality_check_qty', groups='quality.group_quality_user')
|
quality_control_point_qty = fields.Integer(compute='_compute_quality_check_qty',
|
||||||
|
groups='quality.group_quality_user')
|
||||||
quality_pass_qty = fields.Integer(compute='_compute_quality_check_qty', groups='quality.group_quality_user')
|
quality_pass_qty = fields.Integer(compute='_compute_quality_check_qty', groups='quality.group_quality_user')
|
||||||
quality_fail_qty = fields.Integer(compute='_compute_quality_check_qty', groups='quality.group_quality_user')
|
quality_fail_qty = fields.Integer(compute='_compute_quality_check_qty', groups='quality.group_quality_user')
|
||||||
|
|
||||||
@@ -394,14 +902,16 @@ class ProductTemplate(models.Model):
|
|||||||
def _compute_quality_check_qty(self):
|
def _compute_quality_check_qty(self):
|
||||||
for product_tmpl in self:
|
for product_tmpl in self:
|
||||||
product_tmpl.quality_fail_qty, product_tmpl.quality_pass_qty = product_tmpl.product_variant_ids._count_quality_checks()
|
product_tmpl.quality_fail_qty, product_tmpl.quality_pass_qty = product_tmpl.product_variant_ids._count_quality_checks()
|
||||||
product_tmpl.quality_control_point_qty = product_tmpl.with_context(active_test=product_tmpl.active).product_variant_ids._count_quality_points()
|
product_tmpl.quality_control_point_qty = product_tmpl.with_context(
|
||||||
|
active_test=product_tmpl.active).product_variant_ids._count_quality_points()
|
||||||
|
|
||||||
def action_see_quality_control_points(self):
|
def action_see_quality_control_points(self):
|
||||||
self.ensure_one()
|
self.ensure_one()
|
||||||
action = self.env["ir.actions.actions"]._for_xml_id("quality_control.quality_point_action")
|
action = self.env["ir.actions.actions"]._for_xml_id("quality_control.quality_point_action")
|
||||||
action['context'] = dict(self.env.context, default_product_ids=self.product_variant_ids.ids)
|
action['context'] = dict(self.env.context, default_product_ids=self.product_variant_ids.ids)
|
||||||
|
|
||||||
domain_in_products_or_categs = ['|', ('product_ids', 'in', self.product_variant_ids.ids), ('product_category_ids', 'parent_of', self.categ_id.ids)]
|
domain_in_products_or_categs = ['|', ('product_ids', 'in', self.product_variant_ids.ids),
|
||||||
|
('product_category_ids', 'parent_of', self.categ_id.ids)]
|
||||||
domain_no_products_and_categs = [('product_ids', '=', False), ('product_category_ids', '=', False)]
|
domain_no_products_and_categs = [('product_ids', '=', False), ('product_category_ids', '=', False)]
|
||||||
action['domain'] = OR([domain_in_products_or_categs, domain_no_products_and_categs])
|
action['domain'] = OR([domain_in_products_or_categs, domain_no_products_and_categs])
|
||||||
return action
|
return action
|
||||||
@@ -412,10 +922,10 @@ class ProductTemplate(models.Model):
|
|||||||
action['context'] = dict(self.env.context, default_product_id=self.product_variant_id.id, create=False)
|
action['context'] = dict(self.env.context, default_product_id=self.product_variant_id.id, create=False)
|
||||||
action['domain'] = [
|
action['domain'] = [
|
||||||
'|',
|
'|',
|
||||||
('product_id', 'in', self.product_variant_ids.ids),
|
('product_id', 'in', self.product_variant_ids.ids),
|
||||||
'&',
|
'&',
|
||||||
('measure_on', '=', 'operation'),
|
('measure_on', '=', 'operation'),
|
||||||
('picking_id.move_ids.product_tmpl_id', '=', self.id),
|
('picking_id.move_ids.product_tmpl_id', '=', self.id),
|
||||||
]
|
]
|
||||||
return action
|
return action
|
||||||
|
|
||||||
@@ -423,7 +933,8 @@ class ProductTemplate(models.Model):
|
|||||||
class ProductProduct(models.Model):
|
class ProductProduct(models.Model):
|
||||||
_inherit = "product.product"
|
_inherit = "product.product"
|
||||||
|
|
||||||
quality_control_point_qty = fields.Integer(compute='_compute_quality_check_qty', groups='quality.group_quality_user')
|
quality_control_point_qty = fields.Integer(compute='_compute_quality_check_qty',
|
||||||
|
groups='quality.group_quality_user')
|
||||||
quality_pass_qty = fields.Integer(compute='_compute_quality_check_qty', groups='quality.group_quality_user')
|
quality_pass_qty = fields.Integer(compute='_compute_quality_check_qty', groups='quality.group_quality_user')
|
||||||
quality_fail_qty = fields.Integer(compute='_compute_quality_check_qty', groups='quality.group_quality_user')
|
quality_fail_qty = fields.Integer(compute='_compute_quality_check_qty', groups='quality.group_quality_user')
|
||||||
|
|
||||||
@@ -437,10 +948,10 @@ class ProductProduct(models.Model):
|
|||||||
quality_pass_qty = 0
|
quality_pass_qty = 0
|
||||||
domain = [
|
domain = [
|
||||||
'|',
|
'|',
|
||||||
('product_id', 'in', self.ids),
|
('product_id', 'in', self.ids),
|
||||||
'&',
|
'&',
|
||||||
('measure_on', '=', 'operation'),
|
('measure_on', '=', 'operation'),
|
||||||
('picking_id.move_ids.product_id', 'in', self.ids),
|
('picking_id.move_ids.product_id', 'in', self.ids),
|
||||||
('company_id', '=', self.env.company.id),
|
('company_id', '=', self.env.company.id),
|
||||||
('quality_state', '!=', 'none')
|
('quality_state', '!=', 'none')
|
||||||
]
|
]
|
||||||
@@ -464,7 +975,8 @@ class ProductProduct(models.Model):
|
|||||||
_, where_clause, where_clause_args = query.get_sql()
|
_, where_clause, where_clause_args = query.get_sql()
|
||||||
additional_where_clause = self._additional_quality_point_where_clause()
|
additional_where_clause = self._additional_quality_point_where_clause()
|
||||||
where_clause += additional_where_clause
|
where_clause += additional_where_clause
|
||||||
parent_category_ids = [int(parent_id) for parent_id in self.categ_id.parent_path.split('/')[:-1]] if self.categ_id else []
|
parent_category_ids = [int(parent_id) for parent_id in
|
||||||
|
self.categ_id.parent_path.split('/')[:-1]] if self.categ_id else []
|
||||||
|
|
||||||
self.env.cr.execute("""
|
self.env.cr.execute("""
|
||||||
SELECT COUNT(*)
|
SELECT COUNT(*)
|
||||||
@@ -485,7 +997,7 @@ class ProductProduct(models.Model):
|
|||||||
)
|
)
|
||||||
)
|
)
|
||||||
""" % (where_clause,), where_clause_args + [self.ids, parent_category_ids]
|
""" % (where_clause,), where_clause_args + [self.ids, parent_category_ids]
|
||||||
)
|
)
|
||||||
return self.env.cr.fetchone()[0]
|
return self.env.cr.fetchone()[0]
|
||||||
|
|
||||||
def action_see_quality_control_points(self):
|
def action_see_quality_control_points(self):
|
||||||
@@ -493,7 +1005,8 @@ class ProductProduct(models.Model):
|
|||||||
action = self.product_tmpl_id.action_see_quality_control_points()
|
action = self.product_tmpl_id.action_see_quality_control_points()
|
||||||
action['context'].update(default_product_ids=self.ids)
|
action['context'].update(default_product_ids=self.ids)
|
||||||
|
|
||||||
domain_in_products_or_categs = ['|', ('product_ids', 'in', self.ids), ('product_category_ids', 'parent_of', self.categ_id.ids)]
|
domain_in_products_or_categs = ['|', ('product_ids', 'in', self.ids),
|
||||||
|
('product_category_ids', 'parent_of', self.categ_id.ids)]
|
||||||
domain_no_products_and_categs = [('product_ids', '=', False), ('product_category_ids', '=', False)]
|
domain_no_products_and_categs = [('product_ids', '=', False), ('product_category_ids', '=', False)]
|
||||||
action['domain'] = OR([domain_in_products_or_categs, domain_no_products_and_categs])
|
action['domain'] = OR([domain_in_products_or_categs, domain_no_products_and_categs])
|
||||||
return action
|
return action
|
||||||
@@ -504,12 +1017,75 @@ class ProductProduct(models.Model):
|
|||||||
action['context'] = dict(self.env.context, default_product_id=self.id, create=False)
|
action['context'] = dict(self.env.context, default_product_id=self.id, create=False)
|
||||||
action['domain'] = [
|
action['domain'] = [
|
||||||
'|',
|
'|',
|
||||||
('product_id', '=', self.id),
|
('product_id', '=', self.id),
|
||||||
'&',
|
'&',
|
||||||
('measure_on', '=', 'operation'),
|
('measure_on', '=', 'operation'),
|
||||||
('picking_id.move_ids.product_id', '=', self.id),
|
('picking_id.move_ids.product_id', '=', self.id),
|
||||||
]
|
]
|
||||||
return action
|
return action
|
||||||
|
|
||||||
def _additional_quality_point_where_clause(self):
|
def _additional_quality_point_where_clause(self):
|
||||||
return ""
|
return ""
|
||||||
|
|
||||||
|
|
||||||
|
class QualityCheckMeasureLine(models.Model):
|
||||||
|
_name = 'quality.check.measure.line'
|
||||||
|
_description = '质检测量明细'
|
||||||
|
_order = 'sequence, id'
|
||||||
|
|
||||||
|
sequence = fields.Integer('序号')
|
||||||
|
|
||||||
|
check_id = fields.Many2one('quality.check', string='质检单', required=True, ondelete='cascade')
|
||||||
|
|
||||||
|
# 基本信息
|
||||||
|
product_name = fields.Char('产品名称', related='check_id.product_id.name', readonly=True)
|
||||||
|
drawing_no = fields.Char('图号')
|
||||||
|
measure_item = fields.Char('检测项目')
|
||||||
|
|
||||||
|
# 测量值
|
||||||
|
measure_value1 = fields.Char('测量值1')
|
||||||
|
measure_value2 = fields.Char('测量值2')
|
||||||
|
measure_value3 = fields.Char('测量值3')
|
||||||
|
measure_value4 = fields.Char('测量值4')
|
||||||
|
measure_value5 = fields.Char('测量值5')
|
||||||
|
|
||||||
|
# # 展示列数
|
||||||
|
# column_nums = fields.Integer('列数', related='check_id.column_nums')
|
||||||
|
|
||||||
|
# 判定结果
|
||||||
|
measure_result = fields.Selection([
|
||||||
|
('OK', '合格'),
|
||||||
|
('NG', '不合格')
|
||||||
|
], string='判定', default='OK')
|
||||||
|
|
||||||
|
remark = fields.Char('备注')
|
||||||
|
|
||||||
|
def del_measure_value(self):
|
||||||
|
self.ensure_one()
|
||||||
|
self.sudo().unlink()
|
||||||
|
|
||||||
|
|
||||||
|
# 增加出厂检验报告发布历史
|
||||||
|
class QualityCheckReportHistory(models.Model):
|
||||||
|
_name = 'quality.check.report.history'
|
||||||
|
_description = '出厂检验报告发布历史'
|
||||||
|
|
||||||
|
check_id = fields.Many2one('quality.check', string='质检单', required=True, ondelete='cascade')
|
||||||
|
report_number_id = fields.Many2one('documents.document', string='报告编号', readonly=True)
|
||||||
|
|
||||||
|
sequence = fields.Integer('序号')
|
||||||
|
# 操作(发布、撤销发布、重新发布)
|
||||||
|
action = fields.Selection([
|
||||||
|
('publish', '发布'),
|
||||||
|
('cancel_publish', '撤销发布'),
|
||||||
|
('re_publish', '重新发布')
|
||||||
|
], string='操作')
|
||||||
|
# 操作人
|
||||||
|
operator = fields.Char('操作人')
|
||||||
|
# 操作时间
|
||||||
|
operation_time = fields.Datetime('操作时间')
|
||||||
|
# 文档状态(已发布、废弃)
|
||||||
|
document_status = fields.Selection([
|
||||||
|
('published', '已发布'),
|
||||||
|
('canceled', '废弃')
|
||||||
|
], string='操作后文档状态')
|
||||||
|
|||||||
@@ -81,18 +81,38 @@ class StockPicking(models.Model):
|
|||||||
return quality_pickings
|
return quality_pickings
|
||||||
|
|
||||||
def action_cancel(self):
|
def action_cancel(self):
|
||||||
|
"""
|
||||||
|
调拨单取消后,关联取消质量检查单
|
||||||
|
"""
|
||||||
|
context = self.env.context
|
||||||
|
if not context.get('cancel_check_picking') and self.sudo().mapped('check_ids').filtered(
|
||||||
|
lambda x: x.quality_state in ['pass', 'fail']):
|
||||||
|
self.env.cr.rollback()
|
||||||
|
return {
|
||||||
|
'type': 'ir.actions.act_window',
|
||||||
|
'res_model': 'picking.check.cancel.wizard',
|
||||||
|
'name': '取消质检单',
|
||||||
|
'view_mode': 'form',
|
||||||
|
'target': 'new',
|
||||||
|
'context': {
|
||||||
|
'default_picking_id': self.id,
|
||||||
|
'cancel_check_picking': True}
|
||||||
|
}
|
||||||
|
elif self.check_ids.filtered(lambda x: x.quality_state != 'cancel'):
|
||||||
|
self.sudo().mapped('check_ids').filtered(lambda x: x.quality_state != 'cancel').write({
|
||||||
|
'quality_state': 'cancel'
|
||||||
|
})
|
||||||
res = super(StockPicking, self).action_cancel()
|
res = super(StockPicking, self).action_cancel()
|
||||||
self.sudo().mapped('check_ids').filtered(lambda x: x.quality_state == 'none').unlink()
|
# self.sudo().mapped('check_ids').filtered(lambda x: x.quality_state == 'none').unlink()
|
||||||
return res
|
return res
|
||||||
|
|
||||||
def action_open_quality_check_picking(self):
|
def action_open_quality_check_picking(self):
|
||||||
action = self.env["ir.actions.actions"]._for_xml_id("quality_control.quality_check_action_picking")
|
action = self.env["ir.actions.actions"]._for_xml_id("quality_control.quality_check_action_picking")
|
||||||
action['context'] = self.env.context.copy()
|
action['context'] = {
|
||||||
action['context'].update({
|
|
||||||
'search_default_picking_id': [self.id],
|
'search_default_picking_id': [self.id],
|
||||||
'default_picking_id': self.id,
|
'default_picking_id': self.id,
|
||||||
'show_lots_text': self.show_lots_text,
|
'show_lots_text': self.show_lots_text,
|
||||||
})
|
}
|
||||||
return action
|
return action
|
||||||
|
|
||||||
def button_quality_alert(self):
|
def button_quality_alert(self):
|
||||||
|
|||||||
@@ -1,2 +1,7 @@
|
|||||||
id,name,model_id:id,group_id:id,perm_read,perm_write,perm_create,perm_unlink
|
id,name,model_id:id,group_id:id,perm_read,perm_write,perm_create,perm_unlink
|
||||||
access_quality_check_wizard,access.quality_check_wizard,model_quality_check_wizard,quality.group_quality_user,1,1,1,0
|
access_quality_check_wizard,access.quality_check_wizard,model_quality_check_wizard,quality.group_quality_user,1,1,1,0
|
||||||
|
access_quality_check_measure_line,quality.check.measure.line,model_quality_check_measure_line,base.group_user,1,1,1,0
|
||||||
|
access_quality_check_import_complex_model_wizard,quality.check.import.complex.model.wizard,model_quality_check_import_complex_model_wizard,quality.group_quality_user,1,1,1,0
|
||||||
|
access_quality_check_report_history,quality.check.report.history,model_quality_check_report_history,quality.group_quality_user,1,1,1,0
|
||||||
|
access_quality_check_publish_wizard,quality.check.publish.wizard,model_quality_check_publish_wizard,quality.group_quality_user,1,1,1,0
|
||||||
|
access_picking_check_cancel_wizard,access.picking_check_cancel_wizard,model_picking_check_cancel_wizard,quality.group_quality_user,1,1,1,0
|
||||||
|
|||||||
|
BIN
quality_control/static/src/binary/出厂检验报告上传模版.xlsx
Normal file
BIN
quality_control/static/src/binary/出厂检验报告上传模版.xlsx
Normal file
Binary file not shown.
@@ -4,3 +4,11 @@
|
|||||||
min-height: 250px;
|
min-height: 250px;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.measureTableSequence {
|
||||||
|
width: 58px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.measureTable .o_list_table_ungrouped {
|
||||||
|
min-width: auto;
|
||||||
|
}
|
||||||
44
quality_control/views/quality.check.measures.line.xml
Normal file
44
quality_control/views/quality.check.measures.line.xml
Normal file
@@ -0,0 +1,44 @@
|
|||||||
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
|
<odoo>
|
||||||
|
<record id="quality_check_measure_line_tree" model="ir.ui.view">
|
||||||
|
<field name="name">quality.check.measure.line.tree</field>
|
||||||
|
<field name="model">quality.check.measure.line</field>
|
||||||
|
<field name="arch" type="xml">
|
||||||
|
<tree editable="bottom" class="measureTable">
|
||||||
|
<!-- <field name="sequence" class="measureTableSequence"/> -->
|
||||||
|
<!-- <field name="column_nums"/> -->
|
||||||
|
<field name="measure_item"/>
|
||||||
|
<field name="measure_value1" attrs="{ 'column_invisible': [('parent.column_nums', '<', 1)] }"/>
|
||||||
|
<field name="measure_value2" attrs="{ 'column_invisible': [('parent.column_nums', '<', 2)] }"/>
|
||||||
|
<field name="measure_value3" attrs="{ 'column_invisible': [('parent.column_nums', '<', 3)] }"/>
|
||||||
|
<field name="measure_value4" attrs="{ 'column_invisible': [('parent.column_nums', '<', 4)] }"/>
|
||||||
|
<field name="measure_value5" attrs="{ 'column_invisible': [('parent.column_nums', '<', 5)] }"/>
|
||||||
|
<field name="measure_result"/>
|
||||||
|
<field name="remark"/>
|
||||||
|
<button name="del_measure_value" type="object" string="删除" class="btn-danger" attrs="{'invisible': [('parent.publish_status', '=', 'published')]}"/>
|
||||||
|
</tree>
|
||||||
|
</field>
|
||||||
|
</record>
|
||||||
|
|
||||||
|
<record id="quality_check_measure_line_form" model="ir.ui.view">
|
||||||
|
<field name="name">quality.check.measure.line.form</field>
|
||||||
|
<field name="model">quality.check.measure.line</field>
|
||||||
|
<field name="arch" type="xml">
|
||||||
|
<form>
|
||||||
|
<sheet>
|
||||||
|
<group>
|
||||||
|
<group>
|
||||||
|
<field name="measure_item"/>
|
||||||
|
|
||||||
|
<field name="remark"/>
|
||||||
|
</group>
|
||||||
|
<group>
|
||||||
|
<field name="measure_result"/>
|
||||||
|
</group>
|
||||||
|
</group>
|
||||||
|
</sheet>
|
||||||
|
</form>
|
||||||
|
</field>
|
||||||
|
</record>
|
||||||
|
|
||||||
|
</odoo>
|
||||||
@@ -255,6 +255,7 @@
|
|||||||
<field name="quality_state" widget="statusbar"/>
|
<field name="quality_state" widget="statusbar"/>
|
||||||
</header>
|
</header>
|
||||||
<sheet>
|
<sheet>
|
||||||
|
<field name="is_out_check" invisible="1"/>
|
||||||
<div class="oe_button_box" name="button_box">
|
<div class="oe_button_box" name="button_box">
|
||||||
<button name="action_see_alerts" icon="fa-bell" type="object" class="oe_stat_button"
|
<button name="action_see_alerts" icon="fa-bell" type="object" class="oe_stat_button"
|
||||||
attrs="{'invisible': [('alert_count', '=', 0)]}">
|
attrs="{'invisible': [('alert_count', '=', 0)]}">
|
||||||
@@ -264,10 +265,17 @@
|
|||||||
<group>
|
<group>
|
||||||
<group>
|
<group>
|
||||||
<field name="company_id" invisible="1"/>
|
<field name="company_id" invisible="1"/>
|
||||||
|
<field name="categ_type" invisible="1"/>
|
||||||
<field name="product_id" attrs="{'invisible' : [('measure_on', '=', 'operation')]}"/>
|
<field name="product_id" attrs="{'invisible' : [('measure_on', '=', 'operation')]}"/>
|
||||||
<field name="measure_on" attrs="{'readonly': [('point_id', '!=', False)]}"/>
|
<field name="part_name" attrs="{'invisible': [('categ_type', '!=', '成品')], 'readonly': [('publish_status', '=', 'published')]}"/>
|
||||||
<field name="part_name"/>
|
<field name="part_number" attrs="{'invisible': [('categ_type', '!=', '成品')], 'readonly': [('publish_status', '=', 'published')]}"/>
|
||||||
<field name="part_number"/>
|
<field name="material_name" attrs="{'invisible': [('categ_type', '!=', '成品')]}"/>
|
||||||
|
<field name="total_qty" attrs="{'invisible': ['|', ('measure_on', '!=', 'product'), ('is_out_check', '=', False)]}"/>
|
||||||
|
<field name="check_qty" attrs="{'invisible': ['|', ('measure_on', '!=', 'product'), ('is_out_check', '=', False)], 'readonly': [('publish_status', '=', 'published')]}"/>
|
||||||
|
<!-- <field name="categ_type"/> -->
|
||||||
|
<field name="report_number_id" attrs="{'invisible': ['|', ('measure_on', '!=', 'product'), ('is_out_check', '=', False)]}"/>
|
||||||
|
<field name="column_nums" invisible="1"/>
|
||||||
|
<field name="publish_status" invisible="1"/>
|
||||||
<field name="show_lot_text" invisible="1"/>
|
<field name="show_lot_text" invisible="1"/>
|
||||||
<field name="move_line_id" invisible="1"/>
|
<field name="move_line_id" invisible="1"/>
|
||||||
<field name="product_tracking" invisible="1"/>
|
<field name="product_tracking" invisible="1"/>
|
||||||
@@ -300,23 +308,51 @@
|
|||||||
<field name="alert_ids" invisible="1"/>
|
<field name="alert_ids" invisible="1"/>
|
||||||
</group>
|
</group>
|
||||||
<group>
|
<group>
|
||||||
<field name="picking_id"
|
|
||||||
attrs="{'invisible': [('quality_state', 'in', ('pass', 'fail')), ('picking_id', '=', False)]}"/>
|
|
||||||
<field name="point_id"/>
|
<field name="point_id"/>
|
||||||
|
<field name="measure_on" attrs="{'readonly': [('point_id', '!=', False)]}"/>
|
||||||
<field string="Type" name="test_type_id" options="{'no_open': True, 'no_create': True}"
|
<field string="Type" name="test_type_id" options="{'no_open': True, 'no_create': True}"
|
||||||
attrs="{'readonly': [('point_id', '!=', False)]}"/>
|
attrs="{'readonly': [('point_id', '!=', False)]}"/>
|
||||||
|
<field name="picking_id"
|
||||||
|
attrs="{'invisible': [('quality_state', 'in', ('pass', 'fail')), ('picking_id', '=', False)]}"/>
|
||||||
<field name="control_date" invisible="1"/>
|
<field name="control_date" invisible="1"/>
|
||||||
|
<field name="partner_id" string="Partner"
|
||||||
|
attrs="{'invisible': [('partner_id', '=', False)]}"/>
|
||||||
<field name="team_id"/>
|
<field name="team_id"/>
|
||||||
<field name="company_id" groups="base.group_multi_company"/>
|
<field name="company_id" groups="base.group_multi_company"/>
|
||||||
<field name="user_id" string="Control Person" invisible="1"/>
|
<field name="user_id" string="Control Person" invisible="1"/>
|
||||||
<field name="partner_id" string="Partner"
|
<field name="measure_operator" string="操机员" attrs="{'invisible': ['|', ('measure_on', '!=', 'product'), ('is_out_check', '=', False)], 'readonly': [('publish_status', '=', 'published')]}"/>
|
||||||
attrs="{'invisible': [('partner_id', '=', False)]}"/>
|
|
||||||
</group>
|
</group>
|
||||||
</group>
|
</group>
|
||||||
<group attrs="{'invisible': [('test_type', '!=', 'picture')]}">
|
<group attrs="{'invisible': [('test_type', '!=', 'picture')]}">
|
||||||
<field name="picture" widget="image"/>
|
<field name="picture" widget="image"/>
|
||||||
</group>
|
</group>
|
||||||
<notebook>
|
<notebook>
|
||||||
|
<!-- 增加page页:测量、出厂检验报告、2D加工图纸、质检标准、发布历史,它们均在is_out_check为True时显示 -->
|
||||||
|
<!-- 测量page内有一个添加测量值按钮,点击可以新增一行测量值,新增的行在tree视图中显示,显示的列有:序号、检测项目、测量值1、测量值2、测量值3、测量值4、测量值5、判定、备注。其中检测项目、测量值1、测量值2、测量值3、测量值4、测量值5为必填项,判定为下拉框,默认选中合格,备注为文本框。 -->
|
||||||
|
<page string="测量" name="measure" attrs="{'invisible': [('is_out_check', '=', False)]}">
|
||||||
|
<div class="o_row">
|
||||||
|
<button name="add_measure_line" type="object" class="btn-primary" string="添加测量值" attrs="{'invisible': [('publish_status', '=', 'published')]}"/>
|
||||||
|
<button name="remove_measure_line" type="object" class="btn-primary" string="删除测量值" attrs="{'invisible': [('publish_status', '=', 'published')]}"/>
|
||||||
|
<!-- <button name="%(quality_control.import_complex_model_wizard)d" string="上传111" -->
|
||||||
|
<!-- type="action" -->
|
||||||
|
<!-- class="btn-primary" -->
|
||||||
|
<!-- attrs="{'force_show':1, 'invisible': [('publish_status', '=', 'published')]}" -->
|
||||||
|
<!-- context="{'default_model_name': 'quality.check.measure.line', 'default_check_id': id}"/> -->
|
||||||
|
<button name="upload_measure_line" string="上传"
|
||||||
|
type="object"
|
||||||
|
class="btn-primary"
|
||||||
|
attrs="{'force_show':1, 'invisible': [('publish_status', '=', 'published')]}"/>
|
||||||
|
</div>
|
||||||
|
<br/>
|
||||||
|
<div class="o_row">
|
||||||
|
<field name="measure_line_ids" widget="tree" attrs="{'readonly': [('publish_status', '=', 'published')]}"/>
|
||||||
|
</div>
|
||||||
|
</page>
|
||||||
|
<page string="出厂检验报告" name="out_check" attrs="{'invisible': [('is_out_check', '=', False)]}">
|
||||||
|
<field name="report_content" widget="adaptive_viewer"/>
|
||||||
|
</page>
|
||||||
|
|
||||||
<page string="Notes" name="notes">
|
<page string="Notes" name="notes">
|
||||||
<group>
|
<group>
|
||||||
<field name="report_result"/>
|
<field name="report_result"/>
|
||||||
@@ -326,6 +362,18 @@
|
|||||||
|
|
||||||
</group>
|
</group>
|
||||||
</page>
|
</page>
|
||||||
|
<page string="发布历史" name="release_history" attrs="{'invisible': [('is_out_check', '=', False)]}">
|
||||||
|
<field name="report_history_ids">
|
||||||
|
<tree>
|
||||||
|
<field name="sequence"/>
|
||||||
|
<field name="report_number_id"/>
|
||||||
|
<field name="action"/>
|
||||||
|
<field name="document_status"/>
|
||||||
|
<field name="operation_time"/>
|
||||||
|
<field name="operator"/>
|
||||||
|
</tree>
|
||||||
|
</field>
|
||||||
|
</page>
|
||||||
</notebook>
|
</notebook>
|
||||||
</sheet>
|
</sheet>
|
||||||
<div class="oe_chatter">
|
<div class="oe_chatter">
|
||||||
@@ -445,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')]"/>
|
||||||
|
|||||||
@@ -2,3 +2,5 @@
|
|||||||
# Part of Odoo. See LICENSE file for full copyright and licensing details.
|
# Part of Odoo. See LICENSE file for full copyright and licensing details.
|
||||||
|
|
||||||
from . import quality_check_wizard
|
from . import quality_check_wizard
|
||||||
|
from . import import_complex_model
|
||||||
|
from . import quality_wizard
|
||||||
|
|||||||
460
quality_control/wizard/import_complex_model.py
Normal file
460
quality_control/wizard/import_complex_model.py
Normal file
@@ -0,0 +1,460 @@
|
|||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
import base64
|
||||||
|
import logging
|
||||||
|
import os
|
||||||
|
import traceback
|
||||||
|
from datetime import datetime
|
||||||
|
from io import BytesIO
|
||||||
|
from openpyxl import load_workbook
|
||||||
|
import pandas as pd
|
||||||
|
import xlrd
|
||||||
|
|
||||||
|
from odoo import fields, models, api, Command, _
|
||||||
|
# from odoo.exceptions import ValidationError
|
||||||
|
from odoo.exceptions import UserError
|
||||||
|
from datetime import datetime, timedelta
|
||||||
|
|
||||||
|
from odoo.http import request
|
||||||
|
|
||||||
|
_logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
|
class ImportComplexModelWizard(models.TransientModel):
|
||||||
|
_name = 'quality.check.import.complex.model.wizard'
|
||||||
|
file_data = fields.Binary("数据文件")
|
||||||
|
filename = fields.Char(string='文件名')
|
||||||
|
model_name = fields.Char(string='Model Name')
|
||||||
|
field_basis = fields.Char(string='Field Basis')
|
||||||
|
check_id = fields.Many2one(string='质检单', comodel_name='quality.check')
|
||||||
|
|
||||||
|
def get_model_column_name_labels(self, model):
|
||||||
|
fields_info = model.fields_get()
|
||||||
|
# 提取字段名称和展示名称
|
||||||
|
field_labels = {field_info.get('string'): field_info for field_name, field_info in fields_info.items()}
|
||||||
|
|
||||||
|
return field_labels
|
||||||
|
|
||||||
|
def field_name_mapping(selfcolumn, column, field_data):
|
||||||
|
return {}
|
||||||
|
|
||||||
|
@api.model
|
||||||
|
def page_to_field(self, field_data):
|
||||||
|
return {}
|
||||||
|
|
||||||
|
def count_continuous_none(self, df):
|
||||||
|
# 用于存储间断的连续空值的区间
|
||||||
|
none_intervals = []
|
||||||
|
# if pd.isna(val):
|
||||||
|
# 遍历数据并查找连续的 None
|
||||||
|
start = 0
|
||||||
|
num = 0
|
||||||
|
for index, row in df.iterrows():
|
||||||
|
if pd.isna(row[0]):
|
||||||
|
continue
|
||||||
|
else:
|
||||||
|
# 记录区间
|
||||||
|
if num == 0:
|
||||||
|
start = index
|
||||||
|
num = 1
|
||||||
|
else:
|
||||||
|
end = index
|
||||||
|
none_intervals.append({'start': start, 'end': index})
|
||||||
|
start = end
|
||||||
|
# start = None # 重置
|
||||||
|
|
||||||
|
# 检查最后一个区间
|
||||||
|
if len(df) - start >= 1:
|
||||||
|
none_intervals.append({'start': start, 'end': len(df)})
|
||||||
|
|
||||||
|
return none_intervals
|
||||||
|
|
||||||
|
def find_repeated_ranges(self, data):
|
||||||
|
# 判断内联列表范围,有column列的为内联列表字段
|
||||||
|
if not data:
|
||||||
|
return []
|
||||||
|
|
||||||
|
repeats = []
|
||||||
|
start_index = 0
|
||||||
|
current_value = data[0]
|
||||||
|
|
||||||
|
for i in range(1, len(data)):
|
||||||
|
if data[i] == current_value:
|
||||||
|
continue
|
||||||
|
else:
|
||||||
|
if i - start_index > 1: # 有重复
|
||||||
|
repeats.append({'start': start_index, 'end': i, 'column': data[i - 1]})
|
||||||
|
# repeats.append((current_value, start_index, i - 1))
|
||||||
|
current_value = data[i]
|
||||||
|
start_index = i
|
||||||
|
|
||||||
|
# 检查最后一段
|
||||||
|
if len(data) - start_index > 1:
|
||||||
|
repeats.append({'start': start_index, 'end': i, 'column': data[i - 1]})
|
||||||
|
# repeats.append((current_value, start_index, len(data) - 1))
|
||||||
|
|
||||||
|
return repeats
|
||||||
|
|
||||||
|
def convert_float(self, value):
|
||||||
|
if isinstance(value, float) and value.is_integer():
|
||||||
|
return int(value)
|
||||||
|
return value
|
||||||
|
|
||||||
|
def import_data(self):
|
||||||
|
"""导入Excel数据"""
|
||||||
|
if not self.file_data:
|
||||||
|
raise UserError(_('请先上传Excel文件'))
|
||||||
|
|
||||||
|
if self.check_id.measure_line_ids:
|
||||||
|
self.sudo().check_id.measure_line_ids.unlink()
|
||||||
|
|
||||||
|
# 解码文件数据
|
||||||
|
file_content = base64.b64decode(self.file_data)
|
||||||
|
|
||||||
|
try:
|
||||||
|
# 使用xlrd读取Excel文件
|
||||||
|
workbook = xlrd.open_workbook(file_contents=file_content)
|
||||||
|
sheet = workbook.sheet_by_index(0)
|
||||||
|
|
||||||
|
# 检查表头是否符合预期(忽略星号)
|
||||||
|
expected_headers = ['产品名称', '图号', '检测项目', '测量值1', '测量值2', '测量值3', '测量值4', '测量值5', '判定', '备注']
|
||||||
|
actual_headers = sheet.row_values(0)
|
||||||
|
|
||||||
|
# 处理合并单元格的情况
|
||||||
|
# 获取所有合并单元格
|
||||||
|
merged_cells = []
|
||||||
|
for crange in sheet.merged_cells:
|
||||||
|
rlo, rhi, clo, chi = crange
|
||||||
|
if rlo == 0: # 只关注第一行(表头)的合并单元格
|
||||||
|
merged_cells.append((clo, chi))
|
||||||
|
|
||||||
|
# 清理表头(移除星号和处理空值)
|
||||||
|
actual_headers_clean = []
|
||||||
|
for i, header in enumerate(actual_headers):
|
||||||
|
# 检查当前列是否在合并单元格范围内但不是合并单元格的起始列
|
||||||
|
is_merged_not_first = any(clo < i < chi for clo, chi in merged_cells)
|
||||||
|
|
||||||
|
if is_merged_not_first:
|
||||||
|
# 如果是合并单元格的非起始列,跳过
|
||||||
|
continue
|
||||||
|
|
||||||
|
# 处理表头文本
|
||||||
|
if isinstance(header, str):
|
||||||
|
header = header.replace('*', '').strip()
|
||||||
|
|
||||||
|
if header: # 只添加非空表头
|
||||||
|
actual_headers_clean.append(header)
|
||||||
|
|
||||||
|
# 检查表头数量
|
||||||
|
if len(actual_headers_clean) < len(expected_headers):
|
||||||
|
raise UserError(_('表头列数不足,请使用正确的模板文件'))
|
||||||
|
|
||||||
|
# 检查表头顺序(忽略星号)
|
||||||
|
for i, header in enumerate(expected_headers):
|
||||||
|
if i >= len(actual_headers_clean) or header != actual_headers_clean[i]:
|
||||||
|
actual_header = actual_headers_clean[i] if i < len(actual_headers_clean) else "缺失"
|
||||||
|
raise UserError(_('表头顺序不正确,第%s列应为"%s",但实际为"%s"') %
|
||||||
|
(i + 1, header, actual_header))
|
||||||
|
|
||||||
|
# 获取当前活动的quality.check记录
|
||||||
|
active_id = self.env.context.get('active_id')
|
||||||
|
quality_check = self.env['quality.check'].browse(active_id)
|
||||||
|
|
||||||
|
if not quality_check:
|
||||||
|
raise UserError(_('未找到相关的质检记录'))
|
||||||
|
|
||||||
|
# 记录是否有有效数据被导入
|
||||||
|
valid_data_imported = False
|
||||||
|
|
||||||
|
# 从第二行开始读取数据(跳过表头)
|
||||||
|
max_columns = 1
|
||||||
|
for row_index in range(1, sheet.nrows):
|
||||||
|
row = sheet.row_values(row_index)
|
||||||
|
# 检查行是否有数据
|
||||||
|
if not any(row):
|
||||||
|
continue
|
||||||
|
|
||||||
|
if row[2] == '':
|
||||||
|
continue
|
||||||
|
logging.info('================%s, %s==' % (row[1], type(row[1])))
|
||||||
|
|
||||||
|
compare_value = self.convert_float(row[1])
|
||||||
|
if str(compare_value) != quality_check.part_number:
|
||||||
|
print(sheet.row_values(row_index))
|
||||||
|
raise UserError(_('上传内容图号错误,请修改'))
|
||||||
|
for row_index in range(1, sheet.nrows):
|
||||||
|
row = sheet.row_values(row_index)
|
||||||
|
|
||||||
|
# 检查行是否有数据
|
||||||
|
if not any(row):
|
||||||
|
continue
|
||||||
|
|
||||||
|
if row[2] == '':
|
||||||
|
continue
|
||||||
|
|
||||||
|
# 创建quality.check.measure.line记录
|
||||||
|
measure_line_vals = {
|
||||||
|
'check_id': quality_check.id,
|
||||||
|
'sequence': len(quality_check.measure_line_ids) + 1,
|
||||||
|
'product_name': str(row[0]) if row[0] else '', # 产品名称列
|
||||||
|
'drawing_no': str(row[1]) if row[1] else '', # 图号列
|
||||||
|
'measure_item': str(self.convert_float(row[2])) or '', # 检测项目列
|
||||||
|
'measure_value1': str(self.convert_float(row[4])) if row[4] else '', # 测量值1
|
||||||
|
'measure_value2': str(self.convert_float(row[5])) if row[5] else '', # 测量值2
|
||||||
|
'measure_value3': str(self.convert_float(row[6])) if len(row) > 6 and row[6] else '', # 测量值3
|
||||||
|
'measure_value4': str(self.convert_float(row[7])) if len(row) > 7 and row[7] else '', # 测量值4
|
||||||
|
'measure_value5': str(self.convert_float(row[8])) if len(row) > 8 and row[8] else '', # 测量值5
|
||||||
|
'measure_result': 'NG' if row[9] == 'NG' else 'OK', # 判定列
|
||||||
|
'remark': self.convert_float(row[10]) if len(row) > 10 and row[10] else '', # 备注列
|
||||||
|
}
|
||||||
|
|
||||||
|
for i in range(1, 6):
|
||||||
|
if measure_line_vals.get(f'measure_value{i}'):
|
||||||
|
if i > max_columns:
|
||||||
|
max_columns = i
|
||||||
|
|
||||||
|
self.env['quality.check.measure.line'].create(measure_line_vals)
|
||||||
|
valid_data_imported = True
|
||||||
|
|
||||||
|
quality_check.column_nums = max_columns
|
||||||
|
|
||||||
|
# 检查是否有有效数据被导入
|
||||||
|
if not valid_data_imported:
|
||||||
|
raise UserError(_('表格中没有有效数据行可导入,请检查表格内容'))
|
||||||
|
|
||||||
|
# 返回关闭弹窗的动作
|
||||||
|
return {
|
||||||
|
'type': 'ir.actions.act_window_close'
|
||||||
|
}
|
||||||
|
|
||||||
|
except Exception as e:
|
||||||
|
_logger.error("导入Excel数据时出错: %s", str(e))
|
||||||
|
_logger.error(traceback.format_exc())
|
||||||
|
raise UserError(_('导入失败: %s') % str(e))
|
||||||
|
|
||||||
|
def process_first_line(self, df_columns, column_labels, field_data):
|
||||||
|
columns = []
|
||||||
|
last_column_name = None
|
||||||
|
for index, column in enumerate(df_columns):
|
||||||
|
if not column_labels.get(column):
|
||||||
|
if 'Unnamed' in column:
|
||||||
|
columns.append(last_column_name)
|
||||||
|
else:
|
||||||
|
field_name_map = self.page_to_field(field_data)
|
||||||
|
field_name = field_name_map.get(column)
|
||||||
|
if field_name:
|
||||||
|
columns.append(field_name)
|
||||||
|
last_column_name = field_name
|
||||||
|
else:
|
||||||
|
columns.append(column)
|
||||||
|
last_column_name = column
|
||||||
|
else:
|
||||||
|
columns.append(column)
|
||||||
|
last_column_name = column
|
||||||
|
return columns
|
||||||
|
|
||||||
|
def process_inline_list_column(self, columns, first_row, repeat_list):
|
||||||
|
for index, repeat_map in enumerate(repeat_list):
|
||||||
|
start = repeat_map.get('start')
|
||||||
|
end = int(repeat_map.get('end'))
|
||||||
|
if len(repeat_list) - 1 == index:
|
||||||
|
end = end + 1
|
||||||
|
for i in range(start, end):
|
||||||
|
field_name = columns[i]
|
||||||
|
embedded_fields = first_row[i]
|
||||||
|
if pd.isna(embedded_fields):
|
||||||
|
columns[i] = field_name
|
||||||
|
else:
|
||||||
|
columns[i] = field_name + '?' + embedded_fields
|
||||||
|
|
||||||
|
def process_data_list(self, model, data_list, column_labels, sheet):
|
||||||
|
try:
|
||||||
|
for index, data in enumerate(data_list):
|
||||||
|
# 转换每行数据到模型data = {dict: 11} {'刀具物料': '刀片', '刀尖特征': '刀尖倒角', '刀片形状': '五边形', '刀片物料参数': [{'刀片物料参数?内接圆直径IC/D(mm)': 23, '刀片物料参数?刀尖圆弧半径RE(mm)': 2, '刀片物料参数?刀片牙型': '石油管螺纹刀片', '刀片物料参数?切削刃长(mm)': 3, '刀片物料参数?厚度(mm)': 12, '刀片物料参数?后角(mm)': 4, '刀片物料参数?安装孔直径D1(mm)': 2, '刀片物料参数?有无断屑槽': '无', '刀片物料参数?物…视图
|
||||||
|
self.import_model_record(model, data, column_labels, sheet)
|
||||||
|
except Exception as e:
|
||||||
|
traceback_error = traceback.format_exc()
|
||||||
|
logging.error('批量导入失败sheet %s' % sheet)
|
||||||
|
logging.error('批量导入失败 : %s' % traceback_error)
|
||||||
|
raise UserError(e)
|
||||||
|
|
||||||
|
@api.model
|
||||||
|
def process_model_record_data(self, model, data, column_labels, sheet):
|
||||||
|
pass
|
||||||
|
|
||||||
|
def import_model_record(self, model, data, column_labels, sheet):
|
||||||
|
self.process_model_record_data(model, data, column_labels, sheet)
|
||||||
|
model_data = self.convert_column_name(data, column_labels)
|
||||||
|
logging.info('批量导入模型{} 数据: {}'.format(model, model_data))
|
||||||
|
new_model = model.create(model_data)
|
||||||
|
self.model_subsequent_processing(new_model, data)
|
||||||
|
|
||||||
|
@api.model
|
||||||
|
def model_subsequent_processing(self, model, data):
|
||||||
|
pass
|
||||||
|
|
||||||
|
def convert_column_name(self, data, column_labels):
|
||||||
|
tmp_map = {}
|
||||||
|
for key, value in data.items():
|
||||||
|
if not column_labels.get(key):
|
||||||
|
continue
|
||||||
|
if key == "执行标准":
|
||||||
|
print('fqwioiqwfo ', value, column_labels)
|
||||||
|
field_info = column_labels.get(key)
|
||||||
|
tmp_map[field_info.get('name')] = self.process_field_data(value, field_info)
|
||||||
|
return tmp_map
|
||||||
|
|
||||||
|
def process_field_data(self, value, field_info):
|
||||||
|
relation_field_types = ['many2one', 'one2many', 'many2many']
|
||||||
|
field_type = field_info.get('type')
|
||||||
|
if field_type not in relation_field_types:
|
||||||
|
return value
|
||||||
|
if field_type == 'Boolean':
|
||||||
|
if value == '是' or value == '有':
|
||||||
|
return True
|
||||||
|
else:
|
||||||
|
return False
|
||||||
|
if field_type == 'many2one':
|
||||||
|
relation_info = self.env[field_info.get('relation')].sudo().search([('name', '=', value)], limit=1)
|
||||||
|
if relation_info:
|
||||||
|
return int(relation_info)
|
||||||
|
|
||||||
|
if isinstance(value, list):
|
||||||
|
return self.process_basic_data_list(value, field_info)
|
||||||
|
else:
|
||||||
|
relation_info = self.env[field_info.get('relation')].sudo().search([('name', '=', value)], limit=1)
|
||||||
|
if relation_info:
|
||||||
|
return [Command.link(int(relation_info))]
|
||||||
|
|
||||||
|
def process_basic_data_list(self, value, field_info):
|
||||||
|
if self.is_basic_data_list(value):
|
||||||
|
return [
|
||||||
|
Command.link(
|
||||||
|
int(self.env[field_info.get('relation')].sudo().search([('name', '=', element)], limit=1)))
|
||||||
|
for element in value
|
||||||
|
]
|
||||||
|
else:
|
||||||
|
association_column_labels = self.get_model_column_name_labels(
|
||||||
|
self.env[field_info.get('relation')].sudo())
|
||||||
|
data_list = [
|
||||||
|
{column_name.split('?')[1]: column_value
|
||||||
|
for column_name, column_value in association_column_data.items()
|
||||||
|
if
|
||||||
|
len(column_name.split('?')) == 2 and association_column_labels.get(column_name.split('?')[1])}
|
||||||
|
for association_column_data in value
|
||||||
|
]
|
||||||
|
data_list = [self.convert_column_name(element, association_column_labels) for element in data_list]
|
||||||
|
return [
|
||||||
|
Command.create(
|
||||||
|
column_map
|
||||||
|
) for column_map in data_list
|
||||||
|
]
|
||||||
|
|
||||||
|
def get_remaining_ranges(self, ranges, full_list_length):
|
||||||
|
# 创建一个集合用于存储被覆盖的索引
|
||||||
|
covered_indices = set()
|
||||||
|
|
||||||
|
# 遍历范围集合,标记覆盖的索引
|
||||||
|
for r in ranges:
|
||||||
|
start = r['start']
|
||||||
|
end = r['end']
|
||||||
|
# 将该范围内的索引加入集合
|
||||||
|
covered_indices.update(range(start, end + 1))
|
||||||
|
|
||||||
|
# 计算未覆盖的范围
|
||||||
|
remaining_ranges = []
|
||||||
|
start = None
|
||||||
|
|
||||||
|
for i in range(full_list_length):
|
||||||
|
if i not in covered_indices:
|
||||||
|
if start is None:
|
||||||
|
start = i # 开始新的未覆盖范围
|
||||||
|
else:
|
||||||
|
if start is not None:
|
||||||
|
# 记录当前未覆盖范围
|
||||||
|
remaining_ranges.append({'start': start, 'end': i - 1})
|
||||||
|
start = None # 重置
|
||||||
|
|
||||||
|
# 处理最后一个范围
|
||||||
|
if start is not None:
|
||||||
|
remaining_ranges.append({'start': start, 'end': full_list_length - 1})
|
||||||
|
|
||||||
|
return remaining_ranges
|
||||||
|
|
||||||
|
def is_basic_data_list(self, lst):
|
||||||
|
basic_types = (int, float, str, bool)
|
||||||
|
lst_len = len(lst)
|
||||||
|
if lst_len < 1:
|
||||||
|
return False
|
||||||
|
if isinstance(lst[0], basic_types):
|
||||||
|
return True
|
||||||
|
return False
|
||||||
|
|
||||||
|
def index_retrieve_data(self, df, range_map, i, data_map):
|
||||||
|
for remaining_map in range_map:
|
||||||
|
relation_column_data_map = {}
|
||||||
|
for j in range(remaining_map.get('start'), int(remaining_map.get('end')) + 1):
|
||||||
|
value = df.iat[i, j]
|
||||||
|
if pd.isna(value):
|
||||||
|
continue
|
||||||
|
if remaining_map.get('column'):
|
||||||
|
relation_column_data_map.update({df.columns[j]: value})
|
||||||
|
else:
|
||||||
|
if not data_map.get(df.columns[j]):
|
||||||
|
data_map.update({df.columns[j]: value})
|
||||||
|
elif isinstance(data_map.get(df.columns[j]), list):
|
||||||
|
data_map.get(df.columns[j]).append(value)
|
||||||
|
else:
|
||||||
|
lst = [data_map.get(df.columns[j]), value]
|
||||||
|
data_map.update({df.columns[j]: lst})
|
||||||
|
if relation_column_data_map and len(relation_column_data_map) > 0:
|
||||||
|
data_map.setdefault(remaining_map.get('column'), []).append(relation_column_data_map)
|
||||||
|
|
||||||
|
def parse_excel_data_matrix(self, df, repeat_list):
|
||||||
|
row_interval_list = self.count_continuous_none(df)
|
||||||
|
remaining_ranges = self.get_remaining_ranges(repeat_list, len(df.columns))
|
||||||
|
data_list = []
|
||||||
|
for row_interval_map in row_interval_list:
|
||||||
|
data_map = {}
|
||||||
|
for index in range(row_interval_map.get('start'), int(row_interval_map.get('end'))):
|
||||||
|
if index == 0:
|
||||||
|
self.index_retrieve_data(df, remaining_ranges, index, data_map)
|
||||||
|
else:
|
||||||
|
self.index_retrieve_data(df, remaining_ranges, index, data_map)
|
||||||
|
self.index_retrieve_data(df, repeat_list, index, data_map)
|
||||||
|
if len(data_map) > 0:
|
||||||
|
data_list.append(data_map)
|
||||||
|
return data_list
|
||||||
|
|
||||||
|
def saadqw(self):
|
||||||
|
|
||||||
|
excel_template = self.env['excel.template'].sudo().search([('model_id.model', '=', self.model_name)], limit=1)
|
||||||
|
file_content = base64.b64decode(excel_template.file_data)
|
||||||
|
return {
|
||||||
|
'type': 'ir.actions.act_url',
|
||||||
|
'url': 'data:application/vnd.openxmlformats-officedocument.spreadsheetml.sheet;base64,{file_content}',
|
||||||
|
'target': 'self',
|
||||||
|
'download': excel_template.file_name,
|
||||||
|
}
|
||||||
|
# return request.make_response(
|
||||||
|
# file_content,
|
||||||
|
# headers=[
|
||||||
|
# ('Content-Disposition', f'attachment; filename="{excel_template.file_name}"'),
|
||||||
|
# ('Content-Type', 'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet'),
|
||||||
|
# ]
|
||||||
|
# )
|
||||||
|
|
||||||
|
def download_excel_template(self):
|
||||||
|
base_url = self.env['ir.config_parameter'].sudo().get_param(
|
||||||
|
'web.base.url') + '/quality_control/static/src/binary/出厂检验报告上传模版.xlsx'
|
||||||
|
|
||||||
|
# 只有当原始 URL 使用 http 时才替换为 https
|
||||||
|
if base_url.startswith("http://"):
|
||||||
|
excel_url = base_url.replace("http://", "https://")
|
||||||
|
else:
|
||||||
|
excel_url = base_url
|
||||||
|
value = dict(
|
||||||
|
type='ir.actions.act_url',
|
||||||
|
target='self',
|
||||||
|
url=excel_url,
|
||||||
|
)
|
||||||
|
return value
|
||||||
34
quality_control/wizard/import_complex_model.xml
Normal file
34
quality_control/wizard/import_complex_model.xml
Normal file
@@ -0,0 +1,34 @@
|
|||||||
|
<?xml version="1.0" encoding="utf-8" ?>
|
||||||
|
<odoo>
|
||||||
|
<record id="quality_check_import_complex_model_wizard_form" model="ir.ui.view">
|
||||||
|
<field name="name">请导入数据文件</field>
|
||||||
|
<field name="model">quality.check.import.complex.model.wizard</field>
|
||||||
|
<field name="arch" type="xml">
|
||||||
|
<form>
|
||||||
|
<group>
|
||||||
|
<field name="file_data" widget="binary" filename="filename" options="{'accepted_file_extensions': '.xls,.xlsx'}"/>
|
||||||
|
<field name="filename" invisible="1"/>
|
||||||
|
</group>
|
||||||
|
<footer>
|
||||||
|
<button string="确认导入" name="import_data" type="object" class="btn-primary"/>
|
||||||
|
<button string="取消" class="btn-primary" special="cancel"/>
|
||||||
|
<button string="模板文件下载" name="download_excel_template" type="object" class="btn-primary"/>
|
||||||
|
</footer>
|
||||||
|
</form>
|
||||||
|
</field>
|
||||||
|
</record>
|
||||||
|
<record id="import_complex_model_wizard" model="ir.actions.act_window">
|
||||||
|
<field name="name">导入模型数据</field>
|
||||||
|
<field name="type">ir.actions.act_window</field>
|
||||||
|
<!-- <field name="res_model">up.select.wizard</field>-->
|
||||||
|
<field name="res_model">quality.check.import.complex.model.wizard</field>
|
||||||
|
<field name="view_mode">form</field>
|
||||||
|
<field name="view_id" ref="quality_check_import_complex_model_wizard_form"/>
|
||||||
|
<field name="target">new</field>
|
||||||
|
<!-- <field name="context"></field>-->
|
||||||
|
</record>
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
</odoo>
|
||||||
@@ -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')
|
||||||
@@ -62,7 +62,8 @@ class QualityCheckWizard(models.TransientModel):
|
|||||||
|
|
||||||
def do_pass(self):
|
def do_pass(self):
|
||||||
if self.test_type == 'picture' and not self.picture:
|
if self.test_type == 'picture' and not self.picture:
|
||||||
raise UserError('You must provide a picture before validating')
|
raise UserError('请先上传照片')
|
||||||
|
# raise UserError('You must provide a picture before validating')
|
||||||
self.current_check_id.do_pass()
|
self.current_check_id.do_pass()
|
||||||
return self.action_generate_next_window()
|
return self.action_generate_next_window()
|
||||||
|
|
||||||
@@ -112,3 +113,25 @@ class QualityCheckWizard(models.TransientModel):
|
|||||||
default_current_check_id=self.current_check_id.id,
|
default_current_check_id=self.current_check_id.id,
|
||||||
)
|
)
|
||||||
return action
|
return action
|
||||||
|
|
||||||
|
# 对于成品出库的出厂检验报告,增加发布按钮
|
||||||
|
def publish(self):
|
||||||
|
self.current_check_id._check_part_number()
|
||||||
|
self.current_check_id._check_measure_line()
|
||||||
|
self.current_check_id._check_check_qty_and_total_qty()
|
||||||
|
self.current_check_id._do_publish_implementation()
|
||||||
|
|
||||||
|
|
||||||
|
class PickingCheckCancelWizard(models.TransientModel):
|
||||||
|
_name = 'picking.check.cancel.wizard'
|
||||||
|
_description = 'picking check cancel wizard'
|
||||||
|
|
||||||
|
picking_id = fields.Many2one('stock.picking', 'stock picking')
|
||||||
|
|
||||||
|
def confirm_picking_check(self):
|
||||||
|
res = self.picking_id.action_cancel()
|
||||||
|
return res
|
||||||
|
|
||||||
|
def cancel_picking_check(self):
|
||||||
|
# 这里是取消后的逻辑
|
||||||
|
return {'type': 'ir.actions.act_window_close'}
|
||||||
|
|||||||
@@ -79,7 +79,10 @@
|
|||||||
attrs="{'invisible': ['|', ('quality_state', '!=', 'none'), ('test_type', '!=', 'passfail')]}" data-hotkey="x"/>
|
attrs="{'invisible': ['|', ('quality_state', '!=', 'none'), ('test_type', '!=', 'passfail')]}" data-hotkey="x"/>
|
||||||
<button name="action_generate_previous_window" type="object" class="btn-secondary" string="Previous" attrs="{'invisible': [('position_current_check', '=', 1)]}"/>
|
<button name="action_generate_previous_window" type="object" class="btn-secondary" string="Previous" attrs="{'invisible': [('position_current_check', '=', 1)]}"/>
|
||||||
<button name="action_generate_next_window" type="object" class="btn-secondary" string="Next" attrs="{'invisible': [('is_last_check', '=', True)]}"/>
|
<button name="action_generate_next_window" type="object" class="btn-secondary" string="Next" attrs="{'invisible': [('is_last_check', '=', True)]}"/>
|
||||||
|
<button string="发布" name="publish" type="object" class="btn-primary"
|
||||||
|
attrs="{'invisible': ['|', ('quality_state', '!=', 'none'), ('test_type', '!=', 'factory_inspection')]}" data-hotkey="p"/>
|
||||||
<button string="Cancel" class="btn btn-secondary" special="cancel" data-hotkey="z" />
|
<button string="Cancel" class="btn btn-secondary" special="cancel" data-hotkey="z" />
|
||||||
|
|
||||||
</footer>
|
</footer>
|
||||||
</form>
|
</form>
|
||||||
</field>
|
</field>
|
||||||
@@ -118,4 +121,21 @@
|
|||||||
<field name="context">{}</field>
|
<field name="context">{}</field>
|
||||||
<field name="target">new</field>
|
<field name="target">new</field>
|
||||||
</record>
|
</record>
|
||||||
|
|
||||||
|
|
||||||
|
<!-- ================================================================================================== -->
|
||||||
|
<record id="picking_check_cancel_wizard_form" model="ir.ui.view">
|
||||||
|
<field name="name">picking.check.cancel.wizard</field>
|
||||||
|
<field name="model">picking.check.cancel.wizard</field>
|
||||||
|
<field name="arch" type="xml">
|
||||||
|
<form string="Quality Check Failed">
|
||||||
|
<div>质量检查单已完成,继续取消吗?</div>
|
||||||
|
<div class="'color': 'red'">注意:关联质量检查单也将被取消。</div>
|
||||||
|
<footer>
|
||||||
|
<button name="confirm_picking_check" type="object" class="btn-primary" string="确认"/>
|
||||||
|
<button name="cancel_picking_check" type="object" string="取消"/>
|
||||||
|
</footer>
|
||||||
|
</form>
|
||||||
|
</field>
|
||||||
|
</record>
|
||||||
</odoo>
|
</odoo>
|
||||||
|
|||||||
20
quality_control/wizard/quality_wizard.py
Normal file
20
quality_control/wizard/quality_wizard.py
Normal file
@@ -0,0 +1,20 @@
|
|||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
from odoo import api, fields, models, _
|
||||||
|
|
||||||
|
class QualityCheckPublishWizard(models.TransientModel):
|
||||||
|
_name = 'quality.check.publish.wizard'
|
||||||
|
_description = '质检报告发布确认'
|
||||||
|
|
||||||
|
check_id = fields.Many2one('quality.check', string='质检单', required=True)
|
||||||
|
product_name = fields.Char('产品名称', readonly=True)
|
||||||
|
total_qty = fields.Char('总数量', readonly=True)
|
||||||
|
check_qty = fields.Char('检验数', readonly=True)
|
||||||
|
measure_count = fields.Integer('测量件数', readonly=True)
|
||||||
|
item_count = fields.Integer('检验项目数', readonly=True)
|
||||||
|
old_report_name = fields.Char('旧出厂检验报告编号', readonly=True)
|
||||||
|
publish_status = fields.Selection([('draft', '草稿'), ('published', '已发布'), ('canceled', '已撤销')], string='发布状态', readonly=True)
|
||||||
|
|
||||||
|
def action_confirm_publish(self):
|
||||||
|
"""确认发布"""
|
||||||
|
self.ensure_one()
|
||||||
|
return self.check_id._do_publish_implementation()
|
||||||
37
quality_control/wizard/quality_wizard_view.xml
Normal file
37
quality_control/wizard/quality_wizard_view.xml
Normal file
@@ -0,0 +1,37 @@
|
|||||||
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
|
<odoo>
|
||||||
|
<record id="view_quality_check_publish_wizard_form" model="ir.ui.view">
|
||||||
|
<field name="name">quality.check.publish.wizard.form</field>
|
||||||
|
<field name="model">quality.check.publish.wizard</field>
|
||||||
|
<field name="arch" type="xml">
|
||||||
|
<form string="发布确认">
|
||||||
|
<div class="alert alert-info" role="alert" style="margin-bottom:0px;">
|
||||||
|
<p>您即将发布出厂检验报告:产品<strong><field name="product_name" class="oe_inline"/></strong>,总数量<strong><field name="total_qty" class="oe_inline"/></strong>,检验数<strong><field name="check_qty" class="oe_inline"/></strong>,测量<strong><field name="measure_count" class="oe_inline"/></strong>件,检验项目<strong><field name="item_count" class="oe_inline"/></strong>项。</p>
|
||||||
|
<field name="publish_status" invisible="1"/>
|
||||||
|
<!-- 状态为draft时显示 -->
|
||||||
|
<div attrs="{'invisible': [('publish_status', '!=', 'draft')]}">
|
||||||
|
<span style="font-weight:bold;">
|
||||||
|
注意:发布后所有用户可扫描下载本报告
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
<!-- 状态不为draft时显示 -->
|
||||||
|
<div attrs="{'invisible': [('publish_status', '=', 'draft')]}">
|
||||||
|
<span style="font-weight:bold;">
|
||||||
|
注意:已发布的报告
|
||||||
|
<field name="old_report_name" readonly="1"
|
||||||
|
style="color:red;"
|
||||||
|
attrs="{'invisible': [('old_report_name', '=', False)]}"/>
|
||||||
|
<span style="color:red;"
|
||||||
|
attrs="{'invisible': [('old_report_name', '!=', False)]}">(未知报告编号)</span>
|
||||||
|
可能已被客户下载
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<footer>
|
||||||
|
<button name="action_confirm_publish" string="发布" type="object" class="btn-primary"/>
|
||||||
|
<button string="取消" class="btn-secondary" special="cancel"/>
|
||||||
|
</footer>
|
||||||
|
</form>
|
||||||
|
</field>
|
||||||
|
</record>
|
||||||
|
</odoo>
|
||||||
@@ -65,7 +65,7 @@
|
|||||||
<field name="lot_id" position="after">
|
<field name="lot_id" position="after">
|
||||||
<field name="workorder_id" invisible="1"/>
|
<field name="workorder_id" invisible="1"/>
|
||||||
<field name="production_id" invisible="1"/>
|
<field name="production_id" invisible="1"/>
|
||||||
<field name="finished_lot_id" attrs="{'invisible': [('finished_lot_id', '=', False)]}" groups="stock.group_production_lot"/>
|
<!-- <field name="finished_lot_id" attrs="{'invisible': [('finished_lot_id', '=', False)]}" groups="stock.group_production_lot"/> -->
|
||||||
</field>
|
</field>
|
||||||
<xpath expr="//field[@name='lot_id']" position="after">
|
<xpath expr="//field[@name='lot_id']" position="after">
|
||||||
<field name="lot_id" attrs="{'invisible': [('workorder_id', '=', False)]}" groups="stock.group_production_lot" string="Component Lot/Serial"/>
|
<field name="lot_id" attrs="{'invisible': [('workorder_id', '=', False)]}" groups="stock.group_production_lot" string="Component Lot/Serial"/>
|
||||||
|
|||||||
@@ -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': [
|
||||||
],
|
],
|
||||||
|
|||||||
@@ -9,7 +9,6 @@ class Printer(models.Model):
|
|||||||
ip_address = fields.Char(string='IP 地址', required=True)
|
ip_address = fields.Char(string='IP 地址', required=True)
|
||||||
port = fields.Integer(string='端口', default=9100)
|
port = fields.Integer(string='端口', default=9100)
|
||||||
|
|
||||||
|
|
||||||
class TableStyle(models.Model):
|
class TableStyle(models.Model):
|
||||||
_name = 'table.style'
|
_name = 'table.style'
|
||||||
_description = '标签样式'
|
_description = '标签样式'
|
||||||
|
|||||||
@@ -2,7 +2,16 @@
|
|||||||
import time, datetime
|
import time, datetime
|
||||||
import hashlib
|
import hashlib
|
||||||
from odoo import models
|
from odoo import models
|
||||||
|
from typing import Optional
|
||||||
import socket
|
import socket
|
||||||
|
import os
|
||||||
|
import logging
|
||||||
|
import qrcode
|
||||||
|
from reportlab.pdfgen import canvas
|
||||||
|
from reportlab.lib.units import inch
|
||||||
|
from PyPDF2 import PdfFileReader, PdfFileWriter
|
||||||
|
from reportlab.pdfbase import pdfmetrics
|
||||||
|
from reportlab.pdfbase.ttfonts import TTFont
|
||||||
|
|
||||||
class Common(models.Model):
|
class Common(models.Model):
|
||||||
_name = 'sf.sync.common'
|
_name = 'sf.sync.common'
|
||||||
@@ -92,3 +101,145 @@ class PrintingUtils(models.AbstractModel):
|
|||||||
# host = "192.168.50.110" # 可以作为参数传入,或者在此配置
|
# host = "192.168.50.110" # 可以作为参数传入,或者在此配置
|
||||||
# port = 9100 # 可以作为参数传入,或者在此配置
|
# port = 9100 # 可以作为参数传入,或者在此配置
|
||||||
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,
|
||||||
|
qr_code_buttom_text:Optional[str]=False,
|
||||||
|
buttom_text:Optional[str]=False,
|
||||||
|
):
|
||||||
|
"""
|
||||||
|
在PDF文件中添加二维码
|
||||||
|
:param pdf_path: PDF文件路径
|
||||||
|
:param content: 二维码内容
|
||||||
|
:param qr_code_buttom_text: 二维码下方文字
|
||||||
|
:param buttom_text: 正文下方文字
|
||||||
|
:return: 是否成功
|
||||||
|
"""
|
||||||
|
if not os.path.exists(pdf_path):
|
||||||
|
logging.warning(f'PDF文件不存在: {pdf_path}')
|
||||||
|
return False
|
||||||
|
|
||||||
|
# 生成二维码
|
||||||
|
qr = qrcode.QRCode(version=1, box_size=10, border=5)
|
||||||
|
qr.add_data(str(content))
|
||||||
|
qr.make(fit=True)
|
||||||
|
qr_img = qr.make_image(fill_color="black", back_color="white")
|
||||||
|
|
||||||
|
# 保存二维码为临时文件
|
||||||
|
qr_temp_path = '/tmp/qr_temp.png'
|
||||||
|
qr_img.save(qr_temp_path)
|
||||||
|
|
||||||
|
# 创建一个临时PDF文件路径
|
||||||
|
output_temp_path = '/tmp/output_temp.pdf'
|
||||||
|
|
||||||
|
try:
|
||||||
|
# 使用reportlab创建一个新的PDF
|
||||||
|
|
||||||
|
|
||||||
|
# 注册中文字体
|
||||||
|
font_paths = [
|
||||||
|
"/usr/share/fonts/chinese/simsun.ttc", # Windows系统宋体
|
||||||
|
"c:/windows/fonts/simsun.ttc", # Windows系统宋体另一个位置
|
||||||
|
"/usr/share/fonts/truetype/droid/DroidSansFallbackFull.ttf", # Linux Droid字体
|
||||||
|
"/usr/share/fonts/truetype/wqy/wqy-zenhei.ttc", # 文泉驿正黑
|
||||||
|
"/usr/share/fonts/chinese/TrueType/simsun.ttc", # 某些Linux发行版位置
|
||||||
|
]
|
||||||
|
|
||||||
|
font_found = False
|
||||||
|
for font_path in font_paths:
|
||||||
|
if os.path.exists(font_path):
|
||||||
|
try:
|
||||||
|
pdfmetrics.registerFont(TTFont('SimSun', font_path))
|
||||||
|
font_found = True
|
||||||
|
break
|
||||||
|
except:
|
||||||
|
continue
|
||||||
|
|
||||||
|
# 读取原始PDF
|
||||||
|
with open(pdf_path, "rb") as original_file:
|
||||||
|
existing_pdf = PdfFileReader(original_file)
|
||||||
|
output = PdfFileWriter()
|
||||||
|
|
||||||
|
# 处理最后一页
|
||||||
|
last_page = existing_pdf.getNumPages() - 1
|
||||||
|
page = existing_pdf.getPage(last_page)
|
||||||
|
# 获取页面尺寸
|
||||||
|
page_width = float(page.mediaBox.getWidth())
|
||||||
|
page_height = float(page.mediaBox.getHeight())
|
||||||
|
|
||||||
|
# 创建一个新的PDF页面用于放置二维码
|
||||||
|
c = canvas.Canvas(output_temp_path, pagesize=(page_width, page_height))
|
||||||
|
|
||||||
|
# 设置字体
|
||||||
|
if font_found:
|
||||||
|
c.setFont('SimSun', 10) # 增大字体大小到14pt
|
||||||
|
else:
|
||||||
|
# 如果没有找到中文字体,使用默认字体
|
||||||
|
c.setFont('Helvetica', 10)
|
||||||
|
logging.warning("未找到中文字体,将使用默认字体")
|
||||||
|
|
||||||
|
# 在右下角绘制二维码,预留边距
|
||||||
|
qr_size = 1.5 * inch # 二维码大小为2英寸
|
||||||
|
margin = 0.1 * inch # 边距为0.4英寸
|
||||||
|
qr_y = margin + 20 # 将二维码向上移动一点,为文字留出空间
|
||||||
|
c.drawImage(qr_temp_path, page_width - qr_size - margin, qr_y, width=qr_size, height=qr_size)
|
||||||
|
|
||||||
|
if qr_code_buttom_text:
|
||||||
|
# 在二维码下方绘制文字
|
||||||
|
text = qr_code_buttom_text
|
||||||
|
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_y = margin + 20 # 文字位置靠近底部
|
||||||
|
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()
|
||||||
|
|
||||||
|
# 读取带有二维码的临时PDF
|
||||||
|
with open(output_temp_path, "rb") as qr_file:
|
||||||
|
qr_pdf = PdfFileReader(qr_file)
|
||||||
|
qr_page = qr_pdf.getPage(0)
|
||||||
|
|
||||||
|
# 合并原始页面和二维码页面
|
||||||
|
page.mergePage(qr_page)
|
||||||
|
|
||||||
|
# 添加剩余的页面
|
||||||
|
for i in range(0, last_page):
|
||||||
|
output.addPage(existing_pdf.getPage(i))
|
||||||
|
|
||||||
|
output.addPage(page)
|
||||||
|
|
||||||
|
# 保存最终的PDF到一个临时文件
|
||||||
|
final_temp_path = pdf_path + '.tmp'
|
||||||
|
with open(final_temp_path, "wb") as output_file:
|
||||||
|
output.write(output_file)
|
||||||
|
|
||||||
|
# 替换原始文件
|
||||||
|
os.replace(final_temp_path, pdf_path)
|
||||||
|
|
||||||
|
return True
|
||||||
|
|
||||||
|
finally:
|
||||||
|
# 清理临时文件
|
||||||
|
if os.path.exists(qr_temp_path):
|
||||||
|
os.remove(qr_temp_path)
|
||||||
|
if os.path.exists(output_temp_path):
|
||||||
|
os.remove(output_temp_path)
|
||||||
@@ -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
|
||||||
@@ -421,3 +421,4 @@ class EmbryoRedundancy(models.Model):
|
|||||||
width = fields.Float('宽度(mm)', required=True)
|
width = fields.Float('宽度(mm)', required=True)
|
||||||
height = fields.Float('高度(mm)', required=True)
|
height = fields.Float('高度(mm)', required=True)
|
||||||
active = fields.Boolean('有效', default=True)
|
active = fields.Boolean('有效', default=True)
|
||||||
|
remark = fields.Char('描述')
|
||||||
|
|||||||
@@ -57,15 +57,42 @@ class MrsMaterialModel(models.Model):
|
|||||||
remark = fields.Text("备注")
|
remark = fields.Text("备注")
|
||||||
gain_way = fields.Selection(
|
gain_way = fields.Selection(
|
||||||
[("自加工", "自加工"), ("外协", "委外加工"), ("采购", "采购")],
|
[("自加工", "自加工"), ("外协", "委外加工"), ("采购", "采购")],
|
||||||
default="", string="获取方式")
|
default="采购", string="获取方式")
|
||||||
supplier_ids = fields.One2many('sf.supplier.sort', 'materials_model_id', string='供应商')
|
supplier_ids = fields.One2many('sf.supplier.sort', 'materials_model_id', string='供应商')
|
||||||
active = fields.Boolean('有效', default=True)
|
active = fields.Boolean('有效', default=True)
|
||||||
|
|
||||||
@api.constrains("gain_way")
|
def write(self, vals):
|
||||||
def _check_supplier_ids(self):
|
res = super(MrsMaterialModel, self).write(vals)
|
||||||
for item in self:
|
if not self.gain_way:
|
||||||
if item.gain_way in ('外协', '采购') and not item.supplier_ids:
|
self.gain_way = '采购'
|
||||||
raise UserError("请添加供应商")
|
if not self.supplier_ids:
|
||||||
|
supplier_id = self.env['res.partner'].search([('name', 'like', '%傲派%')], limit=1)
|
||||||
|
if not supplier_id:
|
||||||
|
supplier_id = self.env['res.partner'].create({
|
||||||
|
'name': '湖南傲派自动化设备有限公司',
|
||||||
|
'supplier_rank':1,
|
||||||
|
})
|
||||||
|
self.supplier_ids = [(0, 0, {'materials_model_id': self.id, 'partner_id': supplier_id.id or False})]
|
||||||
|
return res
|
||||||
|
@api.model
|
||||||
|
def create(self, vals):
|
||||||
|
res = super(MrsMaterialModel, self).create(vals)
|
||||||
|
if not vals.get('supplier_ids'):
|
||||||
|
supplier_id = self.env['res.partner'].search([('name', 'like', '%傲派%')], limit=1)
|
||||||
|
if not supplier_id:
|
||||||
|
supplier_id = self.env['res.partner'].create({
|
||||||
|
'name': '湖南傲派自动化设备有限公司',
|
||||||
|
'supplier_rank': 1,
|
||||||
|
})
|
||||||
|
res.supplier_ids = [(0, 0, {'materials_model_id': res.id, 'partner_id': supplier_id.id or False})]
|
||||||
|
return res
|
||||||
|
else:
|
||||||
|
return res
|
||||||
|
# @api.constrains("gain_way")
|
||||||
|
# def _check_supplier_ids(self):
|
||||||
|
# for item in self:
|
||||||
|
# if item.gain_way in ('外协', '采购') and not item.supplier_ids:
|
||||||
|
# raise UserError("请添加供应商")
|
||||||
|
|
||||||
|
|
||||||
class MrsProductionProcessCategory(models.Model):
|
class MrsProductionProcessCategory(models.Model):
|
||||||
|
|||||||
@@ -32,6 +32,7 @@ class FixtureModel(models.Model):
|
|||||||
multi_mounting_type_id = fields.Many2one('sf.multi_mounting.type', string="联装类型")
|
multi_mounting_type_id = fields.Many2one('sf.multi_mounting.type', string="联装类型")
|
||||||
brand_id = fields.Many2one('sf.machine.brand', string="品牌")
|
brand_id = fields.Many2one('sf.machine.brand', string="品牌")
|
||||||
model_file = fields.Binary(string="图片")
|
model_file = fields.Binary(string="图片")
|
||||||
|
glb_url = fields.Char(string="图片")
|
||||||
status = fields.Boolean('状态')
|
status = fields.Boolean('状态')
|
||||||
active = fields.Boolean('有效', default=False)
|
active = fields.Boolean('有效', default=False)
|
||||||
|
|
||||||
|
|||||||
@@ -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
|
||||||
|
@@ -148,12 +148,17 @@ td.o_required_modifier {
|
|||||||
color: #aaa;
|
color: #aaa;
|
||||||
}
|
}
|
||||||
|
|
||||||
.o_kanban_primary_left {
|
// .o_kanban_primary_left {
|
||||||
display: flex;
|
// display: flex;
|
||||||
flex-direction: row-reverse;
|
// flex-direction: row-reverse;
|
||||||
justify-content: flex-start;
|
// justify-content: flex-start;
|
||||||
|
// }
|
||||||
|
.o_list_button {
|
||||||
|
min-width: 32px;
|
||||||
|
}
|
||||||
|
.o_list_record_remove {
|
||||||
|
padding-left: 0px !important;
|
||||||
}
|
}
|
||||||
|
|
||||||
.diameter:before {
|
.diameter:before {
|
||||||
content:"Ф";
|
content:"Ф";
|
||||||
display:inline;
|
display:inline;
|
||||||
|
|||||||
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>
|
||||||
@@ -645,6 +645,7 @@
|
|||||||
<field name="long"/>
|
<field name="long"/>
|
||||||
<field name="width"/>
|
<field name="width"/>
|
||||||
<field name="height"/>
|
<field name="height"/>
|
||||||
|
<field name="remark"/>
|
||||||
</tree>
|
</tree>
|
||||||
</field>
|
</field>
|
||||||
</record>
|
</record>
|
||||||
|
|||||||
@@ -5,7 +5,7 @@
|
|||||||
<record model="ir.ui.view" id="mrs_production_process_parameter_tree">
|
<record model="ir.ui.view" id="mrs_production_process_parameter_tree">
|
||||||
<field name="model">sf.production.process.parameter</field>
|
<field name="model">sf.production.process.parameter</field>
|
||||||
<field name="arch" type="xml">
|
<field name="arch" type="xml">
|
||||||
<tree string="表面工艺可选参数" create="0" delete="0">
|
<tree string="工艺可选参数" create="0" delete="0">
|
||||||
<field name="code"/>
|
<field name="code"/>
|
||||||
<field name="name"/>
|
<field name="name"/>
|
||||||
<field name="gain_way"/>
|
<field name="gain_way"/>
|
||||||
@@ -15,7 +15,7 @@
|
|||||||
<record model="ir.ui.view" id="mrs_production_process_parameter_form">
|
<record model="ir.ui.view" id="mrs_production_process_parameter_form">
|
||||||
<field name="model">sf.production.process.parameter</field>
|
<field name="model">sf.production.process.parameter</field>
|
||||||
<field name="arch" type="xml">
|
<field name="arch" type="xml">
|
||||||
<form string="表面工艺可选参数" create="0" delete="0" >
|
<form string="工艺可选参数" create="0" delete="0" >
|
||||||
<sheet>
|
<sheet>
|
||||||
<div class="oe_title">
|
<div class="oe_title">
|
||||||
<h1>
|
<h1>
|
||||||
@@ -104,7 +104,7 @@
|
|||||||
<record model="ir.ui.view" id="sf_production_process_category_form">
|
<record model="ir.ui.view" id="sf_production_process_category_form">
|
||||||
<field name="model">sf.production.process.category</field>
|
<field name="model">sf.production.process.category</field>
|
||||||
<field name="arch" type="xml">
|
<field name="arch" type="xml">
|
||||||
<form string="表面工艺类别" create="0" edit="0" delete="1">
|
<form string="工艺类别" create="0" edit="0" delete="1">
|
||||||
<sheet>
|
<sheet>
|
||||||
<div class="oe_title">
|
<div class="oe_title">
|
||||||
<h1>
|
<h1>
|
||||||
@@ -120,7 +120,7 @@
|
|||||||
</group>
|
</group>
|
||||||
</group>
|
</group>
|
||||||
<notebook>
|
<notebook>
|
||||||
<page string="表面工艺">
|
<page string="工艺">
|
||||||
<field name='production_process_ids' widget="ony2many">
|
<field name='production_process_ids' widget="ony2many">
|
||||||
<tree editable="bottom">
|
<tree editable="bottom">
|
||||||
<field name="code" string="编码号" readonly="1" force_save="1"/>
|
<field name="code" string="编码号" readonly="1" force_save="1"/>
|
||||||
@@ -139,7 +139,7 @@
|
|||||||
<record model="ir.ui.view" id="sf_production_process_category_tree">
|
<record model="ir.ui.view" id="sf_production_process_category_tree">
|
||||||
<field name="model">sf.production.process.category</field>
|
<field name="model">sf.production.process.category</field>
|
||||||
<field name="arch" type="xml">
|
<field name="arch" type="xml">
|
||||||
<tree string="表面工艺类别" default_order="sequence, id" create="0" edit="0" delete="1">
|
<tree string="工艺类别" default_order="sequence, id" create="0" edit="0" delete="1">
|
||||||
<field name="sequence" widget="handle" string="序号" readonly="1"/>
|
<field name="sequence" widget="handle" string="序号" readonly="1"/>
|
||||||
<field name="code"/>
|
<field name="code"/>
|
||||||
<field name="name" string="名称"/>
|
<field name="name" string="名称"/>
|
||||||
@@ -163,7 +163,7 @@
|
|||||||
<record model="ir.ui.view" id="sf_production_process_tree">
|
<record model="ir.ui.view" id="sf_production_process_tree">
|
||||||
<field name="model">sf.production.process</field>
|
<field name="model">sf.production.process</field>
|
||||||
<field name="arch" type="xml">
|
<field name="arch" type="xml">
|
||||||
<tree string="表面工艺" create="0" edit="0" delete="0">
|
<tree string="工艺" create="0" edit="0" delete="0">
|
||||||
<field name="sequence" string="加工顺序" readonly="1"/>
|
<field name="sequence" string="加工顺序" readonly="1"/>
|
||||||
<field name="code"/>
|
<field name="code"/>
|
||||||
<field name="name" string="名称"/>
|
<field name="name" string="名称"/>
|
||||||
@@ -175,7 +175,7 @@
|
|||||||
<record model="ir.ui.view" id="sf_production_process_form">
|
<record model="ir.ui.view" id="sf_production_process_form">
|
||||||
<field name="model">sf.production.process</field>
|
<field name="model">sf.production.process</field>
|
||||||
<field name="arch" type="xml">
|
<field name="arch" type="xml">
|
||||||
<form string="表面工艺" create="0" delete="0">
|
<form string="工艺" create="0" delete="0">
|
||||||
<sheet>
|
<sheet>
|
||||||
<div class="oe_title">
|
<div class="oe_title">
|
||||||
<h1>
|
<h1>
|
||||||
@@ -262,13 +262,13 @@
|
|||||||
<group>
|
<group>
|
||||||
<field name="materials_no" readonly="1" force_save="1"/>
|
<field name="materials_no" readonly="1" force_save="1"/>
|
||||||
<field name="gain_way" required="0"/>
|
<field name="gain_way" required="0"/>
|
||||||
<field name="tensile_strength" required="1"/>
|
<field name="density" readonly="1" required="1" class="custom_required"/>
|
||||||
<field name="hardness" required="1"/>
|
|
||||||
<field name="density" readonly="1"/>
|
|
||||||
</group>
|
</group>
|
||||||
<group>
|
<group>
|
||||||
<field name="rough_machining" required="1"/>
|
<field name="rough_machining" required="1"/>
|
||||||
<field name="finish_machining" required="1"/>
|
<field name="finish_machining" required="1"/>
|
||||||
|
<field name="tensile_strength" required="1"/>
|
||||||
|
<field name="hardness" required="1"/>
|
||||||
<field name="need_h" default="false" readonly="1"/>
|
<field name="need_h" default="false" readonly="1"/>
|
||||||
<field name="mf_materia_post" attrs="{'invisible':[('need_h','=',False)]} "
|
<field name="mf_materia_post" attrs="{'invisible':[('need_h','=',False)]} "
|
||||||
readonly="1"/>
|
readonly="1"/>
|
||||||
@@ -297,7 +297,7 @@
|
|||||||
<record model="ir.ui.view" id="sf_materials_model_tree">
|
<record model="ir.ui.view" id="sf_materials_model_tree">
|
||||||
<field name="model">sf.materials.model</field>
|
<field name="model">sf.materials.model</field>
|
||||||
<field name="arch" type="xml">
|
<field name="arch" type="xml">
|
||||||
<tree string="材料型号" delete="0">
|
<tree string="材料型号" delete="0" create="0">
|
||||||
<field name="materials_no"/>
|
<field name="materials_no"/>
|
||||||
<field name="materials_code"/>
|
<field name="materials_code"/>
|
||||||
<field name="name"/>
|
<field name="name"/>
|
||||||
@@ -395,7 +395,7 @@
|
|||||||
<field name="view_mode">tree,form</field>
|
<field name="view_mode">tree,form</field>
|
||||||
</record>
|
</record>
|
||||||
<record id="sf_production_process" model="ir.actions.act_window">
|
<record id="sf_production_process" model="ir.actions.act_window">
|
||||||
<field name="name">表面工艺</field>
|
<field name="name">工艺</field>
|
||||||
<field name="type">ir.actions.act_window</field>
|
<field name="type">ir.actions.act_window</field>
|
||||||
<field name="res_model">sf.production.process</field>
|
<field name="res_model">sf.production.process</field>
|
||||||
<field name="view_mode">tree,form</field>
|
<field name="view_mode">tree,form</field>
|
||||||
@@ -414,13 +414,13 @@
|
|||||||
<!-- </record>-->
|
<!-- </record>-->
|
||||||
|
|
||||||
<record id="sf_production_process_category" model="ir.actions.act_window">
|
<record id="sf_production_process_category" model="ir.actions.act_window">
|
||||||
<field name="name">表面工艺类别</field>
|
<field name="name">工艺类别</field>
|
||||||
<field name="type">ir.actions.act_window</field>
|
<field name="type">ir.actions.act_window</field>
|
||||||
<field name="res_model">sf.production.process.category</field>
|
<field name="res_model">sf.production.process.category</field>
|
||||||
<field name="view_mode">tree,form</field>
|
<field name="view_mode">tree,form</field>
|
||||||
</record>
|
</record>
|
||||||
<record id="mrs_production_process_parameter_action" model="ir.actions.act_window">
|
<record id="mrs_production_process_parameter_action" model="ir.actions.act_window">
|
||||||
<field name="name">表面工艺可选参数</field>
|
<field name="name">工艺可选参数</field>
|
||||||
<field name="type">ir.actions.act_window</field>
|
<field name="type">ir.actions.act_window</field>
|
||||||
<field name="res_model">sf.production.process.parameter</field>
|
<field name="res_model">sf.production.process.parameter</field>
|
||||||
<field name="view_mode">tree,form</field>
|
<field name="view_mode">tree,form</field>
|
||||||
|
|||||||
@@ -158,6 +158,8 @@
|
|||||||
<!-- <field name="upload_model_file" widget="many2many_binary"/>-->
|
<!-- <field name="upload_model_file" widget="many2many_binary"/>-->
|
||||||
<field name="model_file" widget="Viewer3D" string="模型" readonly="1" force_save="1"
|
<field name="model_file" widget="Viewer3D" string="模型" readonly="1" force_save="1"
|
||||||
attrs="{'invisible': [('model_file', '=', False)]}"/>
|
attrs="{'invisible': [('model_file', '=', False)]}"/>
|
||||||
|
<field name="glb_url" widget="Viewer3D" string="模型" readonly="1" force_save="1"
|
||||||
|
attrs="{'invisible': [('glb_url', '=', False)]}"/>
|
||||||
</group>
|
</group>
|
||||||
</group>
|
</group>
|
||||||
<notebook>
|
<notebook>
|
||||||
|
|||||||
@@ -78,7 +78,7 @@
|
|||||||
|
|
||||||
<menuitem
|
<menuitem
|
||||||
id="menu_sf_production_process"
|
id="menu_sf_production_process"
|
||||||
name="表面工艺"
|
name="工艺"
|
||||||
parent="menu_sf_production_process_1"
|
parent="menu_sf_production_process_1"
|
||||||
sequence="2"
|
sequence="2"
|
||||||
action="sf_production_process"
|
action="sf_production_process"
|
||||||
@@ -86,7 +86,7 @@
|
|||||||
|
|
||||||
<menuitem
|
<menuitem
|
||||||
id="menu_sf_production_process_category"
|
id="menu_sf_production_process_category"
|
||||||
name="表面工艺类别"
|
name="工艺类别"
|
||||||
parent="menu_sf_production_process_1"
|
parent="menu_sf_production_process_1"
|
||||||
sequence="1"
|
sequence="1"
|
||||||
action="sf_production_process_category"
|
action="sf_production_process_category"
|
||||||
@@ -113,7 +113,7 @@
|
|||||||
|
|
||||||
<menuitem
|
<menuitem
|
||||||
id="mrs_production_process_parameter_view"
|
id="mrs_production_process_parameter_view"
|
||||||
name="表面工艺可选参数"
|
name="工艺可选参数"
|
||||||
parent="menu_sf_production_process_1"
|
parent="menu_sf_production_process_1"
|
||||||
sequence="2"
|
sequence="2"
|
||||||
action="mrs_production_process_parameter_action"
|
action="mrs_production_process_parameter_action"
|
||||||
|
|||||||
@@ -29,7 +29,7 @@ class Sf_Bf_Connect(http.Controller):
|
|||||||
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['delivery_end_date'], kw['payments_way'], kw['pay_way'], model_display_version=kw.get('model_display_version'))
|
||||||
i = 1
|
i = 1
|
||||||
# 给sale_order的default_code字段赋值
|
# 给sale_order的default_code字段赋值
|
||||||
aa = request.env['sale.order'].sudo().search([('name', '=', order_id.name)])
|
aa = request.env['sale.order'].sudo().search([('name', '=', order_id.name)])
|
||||||
|
|||||||
@@ -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')
|
||||||
|
|||||||
@@ -32,7 +32,7 @@
|
|||||||
<field name="is_bill" invisible="True"/>
|
<field name="is_bill" invisible="True"/>
|
||||||
<field name="logistics_status" invisible="True"/>
|
<field name="logistics_status" invisible="True"/>
|
||||||
<field name="logistics_way" invisible="True"/>
|
<field name="logistics_way" invisible="True"/>
|
||||||
<button string="物流下单" name="create_order" type="object" confirm="是否确认物流下单" class="btn-primary"
|
<!-- <button string="物流下单" name="create_order" type="object" confirm="是否确认物流下单" class="btn-primary" -->
|
||||||
attrs="{'invisible': ['|', '|', '|', ('check_out', '!=', 'OUT'), ('state', '!=', 'assigned'), ('is_bill', '=', True), ('logistics_way', '=', '自提')]}"/>
|
attrs="{'invisible': ['|', '|', '|', ('check_out', '!=', 'OUT'), ('state', '!=', 'assigned'), ('is_bill', '=', True), ('logistics_way', '=', '自提')]}"/>
|
||||||
<button string="获取物流面单" name="get_bill" type="object" confirm="是否获取物流面单" class="btn-primary"
|
<button string="获取物流面单" name="get_bill" type="object" confirm="是否获取物流面单" class="btn-primary"
|
||||||
attrs="{'invisible': ['|', '|', '|', '|', ('check_out', '!=', 'OUT'), ('state', '!=', 'assigned'), ('logistics_status', '=', '2'), ('is_bill', '=', False), ('logistics_way', '=', '自提')]}"/>
|
attrs="{'invisible': ['|', '|', '|', '|', ('check_out', '!=', 'OUT'), ('state', '!=', 'assigned'), ('logistics_status', '=', '2'), ('is_bill', '=', False), ('logistics_way', '=', '自提')]}"/>
|
||||||
|
|||||||
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
|
||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user