Compare commits
883 Commits
rohitwaghc
...
fix-get_si
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
0255642deb | ||
|
|
2178fdc65a | ||
|
|
eabf706f37 | ||
|
|
d4d9e6c6fa | ||
|
|
2eb318a5ce | ||
|
|
b897225136 | ||
|
|
bb18ae82cf | ||
|
|
74ceb6da5e | ||
|
|
e0ad52b500 | ||
|
|
8252d92e6a | ||
|
|
b15795392b | ||
|
|
274c65c451 | ||
|
|
1b4fbbb115 | ||
|
|
f95403d1dc | ||
|
|
2d2ff7cf52 | ||
|
|
d32a407e08 | ||
|
|
5e0d017497 | ||
|
|
ec07b42ea2 | ||
|
|
3b861798e6 | ||
|
|
b498094a97 | ||
|
|
62bbcbc7ef | ||
|
|
96530b9c0b | ||
|
|
0cbcc55d9a | ||
|
|
6b5c54bcbc | ||
|
|
c2dde04aa2 | ||
|
|
c14986f9e6 | ||
|
|
62a6945bd2 | ||
|
|
be8bb235dc | ||
|
|
498bf15ecd | ||
|
|
210d7711b4 | ||
|
|
cf988434e9 | ||
|
|
466625213b | ||
|
|
9ba6ff67d5 | ||
|
|
eab22eb282 | ||
|
|
7e198ccb21 | ||
|
|
3d54fd8389 | ||
|
|
d0ea598cdf | ||
|
|
6de8c18f98 | ||
|
|
5923b48ede | ||
|
|
7d264696f3 | ||
|
|
b96c063c93 | ||
|
|
2bad706dcf | ||
|
|
7fa3e82ac7 | ||
|
|
7ddbfa10c9 | ||
|
|
4bd437b59d | ||
|
|
fca8028e3c | ||
|
|
cd293a5173 | ||
|
|
07b605a287 | ||
|
|
54383cfb55 | ||
|
|
4cc3c1b765 | ||
|
|
4d56f725fe | ||
|
|
c34f09c503 | ||
|
|
fe43dab4d7 | ||
|
|
b69cdeb4a6 | ||
|
|
15dc5c7e99 | ||
|
|
e05bf9d32a | ||
|
|
3fb7886418 | ||
|
|
bae7c64964 | ||
|
|
68c0e188e8 | ||
|
|
e912e9597d | ||
|
|
610d4f5cb6 | ||
|
|
cdd5441435 | ||
|
|
f09e2130a1 | ||
|
|
0819675fce | ||
|
|
784b6dcfea | ||
|
|
245effcccd | ||
|
|
0d01bd8a5a | ||
|
|
57b6a98703 | ||
|
|
e1b0fffd0c | ||
|
|
c2f88f29dc | ||
|
|
277dade9f9 | ||
|
|
e2cf1ea73e | ||
|
|
df409d80e0 | ||
|
|
5b1571879c | ||
|
|
bb839b2924 | ||
|
|
5bc2035bd0 | ||
|
|
9e973476b2 | ||
|
|
5e2669f4b6 | ||
|
|
0f1be03faf | ||
|
|
351ee5b8fe | ||
|
|
e84c9f7c51 | ||
|
|
7da9ffa3bd | ||
|
|
4415212a2d | ||
|
|
6401908f41 | ||
|
|
beee98da6d | ||
|
|
60c33ac3e6 | ||
|
|
1bc74bde29 | ||
|
|
12c40ef2e4 | ||
|
|
b250a21a2b | ||
|
|
4e28b5a199 | ||
|
|
739434b727 | ||
|
|
9ea963bfe9 | ||
|
|
6e5484ea03 | ||
|
|
d048644327 | ||
|
|
1a9e091d12 | ||
|
|
1b0d9643cd | ||
|
|
92bc962f60 | ||
|
|
a9576f0cf6 | ||
|
|
4f8a16848f | ||
|
|
bacf2b7431 | ||
|
|
34d8bc4701 | ||
|
|
510fdf7bf6 | ||
|
|
0346f47c1d | ||
|
|
3f9693b31f | ||
|
|
877cc7255d | ||
|
|
70abedc57a | ||
|
|
0c6de4ecb2 | ||
|
|
a517125d64 | ||
|
|
b7f283b2f0 | ||
|
|
cd37fd790b | ||
|
|
f983e09f92 | ||
|
|
3b4b2275de | ||
|
|
b5340c5ec0 | ||
|
|
10074e9980 | ||
|
|
bbee9b5637 | ||
|
|
915d864166 | ||
|
|
8555617295 | ||
|
|
b71b0d5997 | ||
|
|
a117ef3cb8 | ||
|
|
33b631e395 | ||
|
|
9b1c22250f | ||
|
|
4feecb69d8 | ||
|
|
12560e2407 | ||
|
|
0890b414b1 | ||
|
|
d54f8318fb | ||
|
|
fe77b9d633 | ||
|
|
47ee801d37 | ||
|
|
705ae7da14 | ||
|
|
7223106417 | ||
|
|
5a5758423e | ||
|
|
b09c9354fb | ||
|
|
d00f6672a8 | ||
|
|
06d6220a2a | ||
|
|
3011322b22 | ||
|
|
e560029736 | ||
|
|
b1ba210332 | ||
|
|
cb9114442b | ||
|
|
eb5bb9f9a9 | ||
|
|
87ba3b64f7 | ||
|
|
47f7b65058 | ||
|
|
161ae1edd1 | ||
|
|
d097ad6c19 | ||
|
|
a5d5223c0e | ||
|
|
1a1629196d | ||
|
|
d0ed8ef83b | ||
|
|
0773f66feb | ||
|
|
2dc49c834a | ||
|
|
787333896c | ||
|
|
283763dfb2 | ||
|
|
07175367d8 | ||
|
|
61219ca4ce | ||
|
|
c20995ec2f | ||
|
|
6a0a08b59c | ||
|
|
d370c60a6c | ||
|
|
a6ab53236e | ||
|
|
9a00edb031 | ||
|
|
ae353398d9 | ||
|
|
9983283f95 | ||
|
|
6d5bdc6c68 | ||
|
|
80e69210db | ||
|
|
793e3ad78e | ||
|
|
4057682c87 | ||
|
|
5e68b7e3a6 | ||
|
|
495b47db16 | ||
|
|
4aa960b744 | ||
|
|
dd6c192695 | ||
|
|
8772628912 | ||
|
|
a99d0a65b0 | ||
|
|
a09241e3c7 | ||
|
|
c7b961ffa2 | ||
|
|
32a608f948 | ||
|
|
8d79365e0d | ||
|
|
71e833c3f2 | ||
|
|
b5f3013005 | ||
|
|
39ef75e2d0 | ||
|
|
42813d38c3 | ||
|
|
55147781f3 | ||
|
|
c9fd182268 | ||
|
|
0743289925 | ||
|
|
4d75159247 | ||
|
|
baa1978128 | ||
|
|
259f313af7 | ||
|
|
863116f1cd | ||
|
|
5cb5e09dbb | ||
|
|
fa1c7b663c | ||
|
|
a9a84cc7d4 | ||
|
|
ff967c45f7 | ||
|
|
9a5c422074 | ||
|
|
5a83a16e60 | ||
|
|
1704180f38 | ||
|
|
0cf9ff0a04 | ||
|
|
41ae2a2dc5 | ||
|
|
f7b2380ec1 | ||
|
|
5eeb650dfd | ||
|
|
a1b95606b1 | ||
|
|
726ac6bda1 | ||
|
|
d6201ce5c7 | ||
|
|
e7e23fbc96 | ||
|
|
e7544e9fc1 | ||
|
|
eaf86a6461 | ||
|
|
f53ba178a8 | ||
|
|
c68ad73c6e | ||
|
|
6851c5042f | ||
|
|
da96578afb | ||
|
|
27f05145ae | ||
|
|
1b3ba25220 | ||
|
|
13cba5068b | ||
|
|
db24e24882 | ||
|
|
e7984b3ef9 | ||
|
|
b562b4cf99 | ||
|
|
69d7a640ee | ||
|
|
fa2d33cb50 | ||
|
|
2588970d55 | ||
|
|
d4ac57704c | ||
|
|
15c90551b6 | ||
|
|
0acd0f50c5 | ||
|
|
acb6e8e120 | ||
|
|
ccff588563 | ||
|
|
780c4278e6 | ||
|
|
2d6506ecec | ||
|
|
a97b3db749 | ||
|
|
cc15f695b4 | ||
|
|
c41cbb3e29 | ||
|
|
6ad298adfc | ||
|
|
db9829e83f | ||
|
|
956c3c50a0 | ||
|
|
40c1acc961 | ||
|
|
7babfd4ac4 | ||
|
|
65df4b6aa8 | ||
|
|
2721ee3a8d | ||
|
|
137b5a6108 | ||
|
|
b023e5d6b3 | ||
|
|
89a0e9c245 | ||
|
|
89326bd657 | ||
|
|
8ddc26eb2e | ||
|
|
ca0c3eb184 | ||
|
|
0465c9aabb | ||
|
|
a045916aca | ||
|
|
be312cea4c | ||
|
|
9611e9bd7f | ||
|
|
08ed3cd313 | ||
|
|
231ab83562 | ||
|
|
f45dd740c5 | ||
|
|
0156339f34 | ||
|
|
f99bb61181 | ||
|
|
6a47a2ceaf | ||
|
|
525f656cc1 | ||
|
|
d34787cf6d | ||
|
|
9424bbc01c | ||
|
|
16c297c2ec | ||
|
|
8beec58670 | ||
|
|
bac811bd5e | ||
|
|
eb9ee3f79b | ||
|
|
5da3e532c9 | ||
|
|
b1d9f3132d | ||
|
|
aaa9036eca | ||
|
|
005c5a587f | ||
|
|
47c78a5a73 | ||
|
|
24ccb3eb78 | ||
|
|
a56b79cc72 | ||
|
|
84ee50e492 | ||
|
|
16b7401d4c | ||
|
|
1a5d56977e | ||
|
|
6812e91893 | ||
|
|
4ca84eadb6 | ||
|
|
27aba02d16 | ||
|
|
3d7ad71b22 | ||
|
|
8d5045ef4c | ||
|
|
00261094c8 | ||
|
|
9471d8fff9 | ||
|
|
ab6e92aae1 | ||
|
|
fb3421fcce | ||
|
|
c8693cdf37 | ||
|
|
3df1d75bdd | ||
|
|
fc12238fcc | ||
|
|
15fff84bb5 | ||
|
|
01aadbef85 | ||
|
|
ccdcb7dfcc | ||
|
|
94fabe0321 | ||
|
|
dd39da0b77 | ||
|
|
3688d9412e | ||
|
|
4918aeb4c6 | ||
|
|
d659d407a0 | ||
|
|
262cafc430 | ||
|
|
11190aac4c | ||
|
|
6d31563920 | ||
|
|
b6957ddac2 | ||
|
|
b2d8a44199 | ||
|
|
2815d196de | ||
|
|
64266c4d38 | ||
|
|
f4d418ea6d | ||
|
|
a008f5f611 | ||
|
|
74eab91042 | ||
|
|
9aeb3932d0 | ||
|
|
b6a7549407 | ||
|
|
9087e1443e | ||
|
|
5952cfa673 | ||
|
|
a1e0197a8b | ||
|
|
af7bc4f178 | ||
|
|
023bc36592 | ||
|
|
ceeb724acc | ||
|
|
5e9016ffab | ||
|
|
6bc40373f2 | ||
|
|
14c8c8c33d | ||
|
|
0925706d5e | ||
|
|
ca14ae8f1b | ||
|
|
90c6d4dc85 | ||
|
|
228aa1a244 | ||
|
|
993e2bfbf9 | ||
|
|
ea3071db66 | ||
|
|
395299803f | ||
|
|
5e2d21c033 | ||
|
|
c4c3090f46 | ||
|
|
6492019383 | ||
|
|
f6831fba13 | ||
|
|
16db6c2f47 | ||
|
|
bd3dc6482e | ||
|
|
f7b7b2b438 | ||
|
|
99fbd8ad18 | ||
|
|
9d20256366 | ||
|
|
85f2a6dd54 | ||
|
|
de2eba0d98 | ||
|
|
705ef4f5a3 | ||
|
|
b2dde55f2c | ||
|
|
d53b34c0ce | ||
|
|
a427029151 | ||
|
|
2de3e6ce6d | ||
|
|
03a38ed025 | ||
|
|
26503a205f | ||
|
|
5a25c80f2e | ||
|
|
4c8a8c3bcd | ||
|
|
decdbd2782 | ||
|
|
eb4c476490 | ||
|
|
68f5dd3e7b | ||
|
|
68aee8c144 | ||
|
|
b24e28953b | ||
|
|
9656412bba | ||
|
|
61280117eb | ||
|
|
961bdf0d24 | ||
|
|
080aa30407 | ||
|
|
eecf9cd1d8 | ||
|
|
e2a519464b | ||
|
|
60a81a563e | ||
|
|
3a66aefd2c | ||
|
|
54f7cf59fd | ||
|
|
ba10f7d04c | ||
|
|
691e3bb24f | ||
|
|
832734ff4c | ||
|
|
3d4156cc7d | ||
|
|
cc053ad894 | ||
|
|
ae294ee470 | ||
|
|
e4bdd3a28d | ||
|
|
63313eef6f | ||
|
|
1da9087cc4 | ||
|
|
de3795a7d4 | ||
|
|
cfd3230c75 | ||
|
|
e38b06bf2d | ||
|
|
7cb0b1b7c3 | ||
|
|
62b4a263f8 | ||
|
|
96b13c59c1 | ||
|
|
1423b38d50 | ||
|
|
ff27cccff4 | ||
|
|
abc7d30024 | ||
|
|
591f4ebdca | ||
|
|
ef8e4191cd | ||
|
|
cf97e3c21f | ||
|
|
2e8739fff7 | ||
|
|
3da0aa6a0b | ||
|
|
97be527ee9 | ||
|
|
cac3b4a0d8 | ||
|
|
96f31847b2 | ||
|
|
85f9f8d176 | ||
|
|
7b0cd03f88 | ||
|
|
68585f6f2b | ||
|
|
857f2b5a01 | ||
|
|
258148b615 | ||
|
|
7145b040f1 | ||
|
|
874766a82f | ||
|
|
096a2c8cd0 | ||
|
|
4a047fefb8 | ||
|
|
fddf341f44 | ||
|
|
fe5fc5bd3a | ||
|
|
0255e09285 | ||
|
|
2b0b15f4d1 | ||
|
|
8e4b591ea2 | ||
|
|
663bb8726c | ||
|
|
2add802d0d | ||
|
|
adfcdb3b65 | ||
|
|
9fadf5f426 | ||
|
|
3e6306348a | ||
|
|
458dd51af7 | ||
|
|
23b0b8ba36 | ||
|
|
592fc81260 | ||
|
|
c232acbe9b | ||
|
|
640dfab827 | ||
|
|
01044ca8e9 | ||
|
|
2633d7dca3 | ||
|
|
ecb533c4d1 | ||
|
|
5fc19dab54 | ||
|
|
58114e7b24 | ||
|
|
1763824e5f | ||
|
|
aee2e12f39 | ||
|
|
500573067a | ||
|
|
8052103197 | ||
|
|
78ab11f991 | ||
|
|
3263f2023c | ||
|
|
8f00481c5f | ||
|
|
9872e371a2 | ||
|
|
37b3ac7952 | ||
|
|
d891bd7fac | ||
|
|
d366a91d9e | ||
|
|
8c3713b649 | ||
|
|
729fc738af | ||
|
|
add238c892 | ||
|
|
ad3634be7c | ||
|
|
5a53a4b044 | ||
|
|
5a97fa6336 | ||
|
|
657bb9d682 | ||
|
|
b9f5a1c85d | ||
|
|
592ce45da7 | ||
|
|
d9b3b95854 | ||
|
|
5426b93387 | ||
|
|
284a40aa63 | ||
|
|
2989e36b1d | ||
|
|
d8245cef72 | ||
|
|
edf4514d8b | ||
|
|
477d9fa87e | ||
|
|
01f133f8c8 | ||
|
|
d32147f8fe | ||
|
|
86aeacf393 | ||
|
|
dd80d3b9b9 | ||
|
|
98cba5ed30 | ||
|
|
ca8a5b45ba | ||
|
|
9d6a06aec5 | ||
|
|
dd016e6ced | ||
|
|
0fe6dcd742 | ||
|
|
bcbe6c4a53 | ||
|
|
f09c5f32cf | ||
|
|
3ca7e442bc | ||
|
|
8ac647ece7 | ||
|
|
4307d3b5c9 | ||
|
|
aa17110bde | ||
|
|
3f6d805033 | ||
|
|
64b44a360a | ||
|
|
040cc8d22f | ||
|
|
3be345e605 | ||
|
|
e1cea25781 | ||
|
|
a2ede7d6d5 | ||
|
|
7e4dd33ab0 | ||
|
|
45d5cff47d | ||
|
|
745e3bfb73 | ||
|
|
762906f240 | ||
|
|
9ec6f1e1d6 | ||
|
|
628ea42b63 | ||
|
|
1efff268b0 | ||
|
|
b206b0583b | ||
|
|
816b1b6bd5 | ||
|
|
24fcd67f8b | ||
|
|
c99c805743 | ||
|
|
f258ab5e98 | ||
|
|
880a85d2af | ||
|
|
efd31a429c | ||
|
|
728cc9f725 | ||
|
|
52305e3000 | ||
|
|
1657337887 | ||
|
|
362f377f61 | ||
|
|
8bdb61cb87 | ||
|
|
67f43d37df | ||
|
|
3543f86c63 | ||
|
|
627165dc7c | ||
|
|
874774fe6c | ||
|
|
ee76af7681 | ||
|
|
f9713eeb56 | ||
|
|
0ca7527f7a | ||
|
|
2a41da94d4 | ||
|
|
73586fd9b2 | ||
|
|
9c889b37fb | ||
|
|
b11ae4b54c | ||
|
|
e3af1dc864 | ||
|
|
8b04c1d4f6 | ||
|
|
74f9e34182 | ||
|
|
1fddc30350 | ||
|
|
e2bb4e2baa | ||
|
|
9006c9b747 | ||
|
|
6518582ed3 | ||
|
|
5c308a4f9a | ||
|
|
ac91030b31 | ||
|
|
9680edfcc3 | ||
|
|
56e991b7f4 | ||
|
|
9903049c7a | ||
|
|
f34ffc2062 | ||
|
|
f50d933a25 | ||
|
|
ce092bf23b | ||
|
|
fb06ad7330 | ||
|
|
80ab4eea8c | ||
|
|
83a13e22b7 | ||
|
|
ee0c64215d | ||
|
|
5ac8bd7f08 | ||
|
|
2f3fc12c08 | ||
|
|
331ad62f3c | ||
|
|
f6e93f084a | ||
|
|
7b2eacd4d8 | ||
|
|
a9bf906545 | ||
|
|
97090ff367 | ||
|
|
6c8b6de4c9 | ||
|
|
f91752cad2 | ||
|
|
cc60c328fe | ||
|
|
ff5b1b7ded | ||
|
|
a1ff7cab7e | ||
|
|
32f622ef80 | ||
|
|
969616ed09 | ||
|
|
ca62cde9aa | ||
|
|
3d1e3a9cde | ||
|
|
184848edf9 | ||
|
|
65ba5b3000 | ||
|
|
3a487bd33a | ||
|
|
56e92b702c | ||
|
|
45299fe4b3 | ||
|
|
434c2a1815 | ||
|
|
f31002636b | ||
|
|
b5dd0c8630 | ||
|
|
9c7b19e0b7 | ||
|
|
3a51a3f37e | ||
|
|
a5a5341643 | ||
|
|
20c6e9fca2 | ||
|
|
5b3eba7bee | ||
|
|
8c6e341a71 | ||
|
|
545ef3c234 | ||
|
|
7842c9fba8 | ||
|
|
7e43d7b131 | ||
|
|
089da459f7 | ||
|
|
b27af6b5c8 | ||
|
|
c69fb80222 | ||
|
|
134201794a | ||
|
|
2df767f596 | ||
|
|
da80e4dbce | ||
|
|
5cc8603cff | ||
|
|
426c245032 | ||
|
|
9731b74ad3 | ||
|
|
ee29526bbe | ||
|
|
3e884d347a | ||
|
|
6e0362dee8 | ||
|
|
2499675ad1 | ||
|
|
908b21f7fd | ||
|
|
e93a19ffb5 | ||
|
|
3e77c0b564 | ||
|
|
e769e750ec | ||
|
|
c31ee8ea33 | ||
|
|
894ae1fe0f | ||
|
|
5fae2f6d57 | ||
|
|
ad5edbb1de | ||
|
|
780b827adc | ||
|
|
8e010ef063 | ||
|
|
09f9764bbd | ||
|
|
a59c942cd4 | ||
|
|
b097bb20d9 | ||
|
|
2f9e96e324 | ||
|
|
56b8d1b927 | ||
|
|
5b446d4575 | ||
|
|
9ae5c979e8 | ||
|
|
ea7565889f | ||
|
|
4a111f7362 | ||
|
|
ffd171a26b | ||
|
|
95edd82638 | ||
|
|
e07c3aad6b | ||
|
|
a393a6b76c | ||
|
|
d380bf8179 | ||
|
|
8062d2be3b | ||
|
|
d17e37c581 | ||
|
|
6f432b8e45 | ||
|
|
94faa44697 | ||
|
|
9fde782403 | ||
|
|
ecc305dd59 | ||
|
|
6f6d5cb4cf | ||
|
|
8634abc021 | ||
|
|
a74e1f1600 | ||
|
|
59438ee8d4 | ||
|
|
4ed9927a30 | ||
|
|
ae508144cd | ||
|
|
a52a1b49af | ||
|
|
93295bf25b | ||
|
|
73639db910 | ||
|
|
c320288690 | ||
|
|
0aa1636d04 | ||
|
|
194d70f8a0 | ||
|
|
af35590549 | ||
|
|
a248b13cc3 | ||
|
|
b0dfc936a1 | ||
|
|
73090fa130 | ||
|
|
ade09bc709 | ||
|
|
8d9f391309 | ||
|
|
ebd74a4e5b | ||
|
|
c4f8f3613f | ||
|
|
e5a8ad54e2 | ||
|
|
815c616f18 | ||
|
|
1d8fcd66e6 | ||
|
|
6ca3b26820 | ||
|
|
18a2e6ecb2 | ||
|
|
922fffda1f | ||
|
|
1fc5844025 | ||
|
|
45b4bfc947 | ||
|
|
92f6d2c87c | ||
|
|
c2bda2c705 | ||
|
|
4b4b176fcf | ||
|
|
a27374fd8f | ||
|
|
56ac3424d2 | ||
|
|
696e2108ac | ||
|
|
89612f2605 | ||
|
|
a8216b9727 | ||
|
|
779260fb49 | ||
|
|
860b67e9c0 | ||
|
|
47ba357bea | ||
|
|
3502c01aa4 | ||
|
|
6e3e094c95 | ||
|
|
a89afb65d7 | ||
|
|
9a171db97f | ||
|
|
48567ef755 | ||
|
|
b57eba6eaf | ||
|
|
70d99eebc0 | ||
|
|
4783e4beee | ||
|
|
450c2470e9 | ||
|
|
09439334ca | ||
|
|
f9c88ea7bc | ||
|
|
33eedb97dc | ||
|
|
2984a86f37 | ||
|
|
5f5d75a0bb | ||
|
|
6f231e4c83 | ||
|
|
c750e4d7ef | ||
|
|
291a499124 | ||
|
|
67b36a0823 | ||
|
|
162c0497d1 | ||
|
|
416bd400bb | ||
|
|
ee60fa940c | ||
|
|
6210b24c64 | ||
|
|
68c6ad6036 | ||
|
|
10b9570429 | ||
|
|
11c8d9fcf1 | ||
|
|
61705047b0 | ||
|
|
ea4b6ff27b | ||
|
|
5ce395a60a | ||
|
|
ac79b8483f | ||
|
|
0135293127 | ||
|
|
adff287160 | ||
|
|
b651b36fff | ||
|
|
5a068410c6 | ||
|
|
ebb186c8df | ||
|
|
d582a73795 | ||
|
|
e845b63228 | ||
|
|
cfe0479dfb | ||
|
|
67e74d03ed | ||
|
|
8722318081 | ||
|
|
2d272fa51c | ||
|
|
15840d408b | ||
|
|
f9fc6c9c9d | ||
|
|
30c6b83a10 | ||
|
|
1754d027b3 | ||
|
|
a1f8595a6a | ||
|
|
5cce522ecd | ||
|
|
758ec720de | ||
|
|
6280031722 | ||
|
|
ff0343d2cc | ||
|
|
3d00d74fed | ||
|
|
30402033bc | ||
|
|
de445b32f5 | ||
|
|
34d3eb88b3 | ||
|
|
a9d91189b0 | ||
|
|
84f0d1ff1f | ||
|
|
98a8288da2 | ||
|
|
0e100cd451 | ||
|
|
e5a018f84c | ||
|
|
2b02ef0066 | ||
|
|
e5bc8fccb1 | ||
|
|
0c5bdbdcf3 | ||
|
|
a3191f1c8c | ||
|
|
787784e937 | ||
|
|
539ff03a7e | ||
|
|
e42a3e0084 | ||
|
|
8fbd4cea5b | ||
|
|
05f24ede96 | ||
|
|
54b323e557 | ||
|
|
9231706227 | ||
|
|
1f88b1ef84 | ||
|
|
10242235bc | ||
|
|
73b65ac82e | ||
|
|
f52916a2c3 | ||
|
|
98d6cdd53c | ||
|
|
e1a87a802d | ||
|
|
d9e284366d | ||
|
|
56e9a46c17 | ||
|
|
2ce6bbf291 | ||
|
|
568d5bfbe8 | ||
|
|
60435daba3 | ||
|
|
f14d1eb871 | ||
|
|
d4c0dbfacc | ||
|
|
469ae2c7f1 | ||
|
|
a9372c42cd | ||
|
|
edf67444ea | ||
|
|
4867ca353c | ||
|
|
4015723591 | ||
|
|
f70d779034 | ||
|
|
23beb46d15 | ||
|
|
e019d43d0b | ||
|
|
c37e374fdd | ||
|
|
cd1e016163 | ||
|
|
1f4b381748 | ||
|
|
fb9a80923b | ||
|
|
ed1c198897 | ||
|
|
1b808e1d7c | ||
|
|
639f427d6d | ||
|
|
539f0251d9 | ||
|
|
d91d9b02d4 | ||
|
|
0104897d69 | ||
|
|
5530a5b303 | ||
|
|
a9fceeb00f | ||
|
|
34e3538b55 | ||
|
|
851a234988 | ||
|
|
17893eff59 | ||
|
|
54e8ce1ac5 | ||
|
|
38e5e4a893 | ||
|
|
65cc804186 | ||
|
|
8fa677b8e8 | ||
|
|
6bd56d2d5f | ||
|
|
204face50d | ||
|
|
c5f5aa8208 | ||
|
|
3d9938221a | ||
|
|
f0a1f4ac7c | ||
|
|
ec1a7869f8 | ||
|
|
7e67d42d1d | ||
|
|
da9a7ff63d | ||
|
|
47f913abcb | ||
|
|
fc4199504f | ||
|
|
0fc14b72ca | ||
|
|
804ed0d26a | ||
|
|
028b3e2fbf | ||
|
|
e0a03789ae | ||
|
|
7a3e4a8a05 | ||
|
|
e16cc38b70 | ||
|
|
c43b30cf36 | ||
|
|
07a8024b45 | ||
|
|
b64147cce9 | ||
|
|
058bb95171 | ||
|
|
daf2ec063c | ||
|
|
77af247450 | ||
|
|
1fd888175f | ||
|
|
fb0ec74d08 | ||
|
|
eb73017798 | ||
|
|
139a68fd0f | ||
|
|
d758fc1b89 | ||
|
|
056b74b162 | ||
|
|
e752f8f0ef | ||
|
|
54a32adef3 | ||
|
|
ca69845238 | ||
|
|
d2fdda8bcd | ||
|
|
afc64ed9ee | ||
|
|
500435b856 | ||
|
|
a15484fe3d | ||
|
|
76c61c1b04 | ||
|
|
de58c67991 | ||
|
|
8a34a4ba68 | ||
|
|
208d5042ee | ||
|
|
ece7049390 | ||
|
|
e72afd0bd6 | ||
|
|
83db7c6a65 | ||
|
|
62384e9321 | ||
|
|
1081df3d7e | ||
|
|
3a8736374c | ||
|
|
f276fbba4f | ||
|
|
fd78f868e1 | ||
|
|
d99a56bc27 | ||
|
|
48c66b68ab | ||
|
|
8d9b90f3f5 | ||
|
|
4a86375e89 | ||
|
|
b60c57a97d | ||
|
|
8903c1bc6f | ||
|
|
e33fd450fd | ||
|
|
dc5d2c7406 | ||
|
|
1612d7ba3f | ||
|
|
681782121c | ||
|
|
35cf944cb7 | ||
|
|
4bbad7f448 | ||
|
|
74b6bfb9eb | ||
|
|
46ea868559 | ||
|
|
3290df5593 | ||
|
|
8e3b9ec879 | ||
|
|
787fc8737f | ||
|
|
fcfcf6957e | ||
|
|
8ffa2bfe25 | ||
|
|
d69b0d76dd | ||
|
|
2bcff4c7f2 | ||
|
|
d436a40739 | ||
|
|
5deba1b6f9 | ||
|
|
886102d462 | ||
|
|
7be578485e | ||
|
|
7c1b990c55 | ||
|
|
74a0d6408a | ||
|
|
92cbe580e6 | ||
|
|
d92eb0c603 | ||
|
|
11d956fa18 | ||
|
|
8e523961dc | ||
|
|
cdbe1c87d4 | ||
|
|
3f42128fff | ||
|
|
4dff2c7a0d | ||
|
|
547993f801 | ||
|
|
946228d783 | ||
|
|
23df4205f8 | ||
|
|
89f484282a | ||
|
|
9ef26e1df0 | ||
|
|
025acc0e48 | ||
|
|
6942ab1012 | ||
|
|
3bfb7b79f2 | ||
|
|
24788ddcc0 | ||
|
|
adf313a6d3 | ||
|
|
7f39318340 | ||
|
|
a432290a82 | ||
|
|
2b64e1ca8b | ||
|
|
514d5434a3 | ||
|
|
7e600a6494 | ||
|
|
17ebc1ea80 | ||
|
|
b099590b2c | ||
|
|
5323bb7bee | ||
|
|
55dbcee36a | ||
|
|
ec9434aae3 | ||
|
|
d1ec0a6093 | ||
|
|
5136fe196b | ||
|
|
4aa841786f | ||
|
|
98cc7434d2 | ||
|
|
35020a9423 | ||
|
|
1cc1c9aa38 | ||
|
|
9d392970f0 | ||
|
|
b0d440c34b | ||
|
|
21c3d9c371 | ||
|
|
dae66eab36 | ||
|
|
9474908449 | ||
|
|
eec4057e8d | ||
|
|
08bc77fb95 | ||
|
|
fff97b1cd2 | ||
|
|
2a7d1c4c8d | ||
|
|
0b674b608b | ||
|
|
14b009b093 | ||
|
|
530922848f | ||
|
|
79d51a0a0b | ||
|
|
85488cd0dc | ||
|
|
ce7ac29d06 | ||
|
|
4f363f5bf3 | ||
|
|
ff7108a3b1 | ||
|
|
77cc91d06b | ||
|
|
45395027d3 | ||
|
|
6d5ccde864 | ||
|
|
961d2d9926 | ||
|
|
75441017c6 | ||
|
|
ed2457bddf | ||
|
|
4471ad581e | ||
|
|
7ecc0d5a04 | ||
|
|
78fe567419 | ||
|
|
5ae9c2f62b | ||
|
|
705dadae8e | ||
|
|
40cdde8820 | ||
|
|
64497c9228 | ||
|
|
188175be84 | ||
|
|
2b4fa98941 | ||
|
|
36a996d704 | ||
|
|
7f1d916f04 | ||
|
|
e15546b42f | ||
|
|
8c61fe2ca5 | ||
|
|
7f865492d0 | ||
|
|
726fba61f3 | ||
|
|
0b85a525fb | ||
|
|
35d92abe73 | ||
|
|
28445058ef | ||
|
|
aab5737ff3 | ||
|
|
115f024260 | ||
|
|
952e8cf60c | ||
|
|
fcfdb9b566 | ||
|
|
ff1dc72d74 | ||
|
|
b3486b43c4 | ||
|
|
f705bf2efe | ||
|
|
6e1565c32c | ||
|
|
4ea43ebc5d | ||
|
|
f292a0cc4c | ||
|
|
4c5a83d6cf | ||
|
|
82774f89b1 | ||
|
|
19b220f39c | ||
|
|
725a7f90e9 |
@@ -9,13 +9,6 @@ trim_trailing_whitespace = true
|
|||||||
charset = utf-8
|
charset = utf-8
|
||||||
|
|
||||||
# python, js indentation settings
|
# python, js indentation settings
|
||||||
[{*.py,*.js,*.vue,*.css,*.scss,*.html}]
|
[{*.py,*.js}]
|
||||||
indent_style = tab
|
indent_style = tab
|
||||||
indent_size = 4
|
indent_size = 4
|
||||||
max_line_length = 110
|
|
||||||
|
|
||||||
# JSON files - mostly doctype schema files
|
|
||||||
[{*.json}]
|
|
||||||
insert_final_newline = false
|
|
||||||
indent_style = space
|
|
||||||
indent_size = 2
|
|
||||||
|
|||||||
@@ -124,7 +124,6 @@
|
|||||||
"beforeEach": true,
|
"beforeEach": true,
|
||||||
"onScan": true,
|
"onScan": true,
|
||||||
"extend_cscript": true,
|
"extend_cscript": true,
|
||||||
"localforage": true,
|
"localforage": true
|
||||||
"Plaid": true
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -28,4 +28,7 @@ b147b85e6ac19a9220cd1e2958a6ebd99373283a
|
|||||||
494bd9ef78313436f0424b918f200dab8fc7c20b
|
494bd9ef78313436f0424b918f200dab8fc7c20b
|
||||||
|
|
||||||
# bulk format python code with black
|
# bulk format python code with black
|
||||||
baec607ff5905b1c67531096a9cf50ec7ff00a5d
|
baec607ff5905b1c67531096a9cf50ec7ff00a5d
|
||||||
|
|
||||||
|
# bulk refactor with sourcery
|
||||||
|
eb9ee3f79b94e594fc6dfa4f6514580e125eee8c
|
||||||
|
|||||||
60
.github/helper/translation.py
vendored
Normal file
60
.github/helper/translation.py
vendored
Normal file
@@ -0,0 +1,60 @@
|
|||||||
|
import re
|
||||||
|
import sys
|
||||||
|
|
||||||
|
errors_encounter = 0
|
||||||
|
pattern = re.compile(r"_\(([\"']{,3})(?P<message>((?!\1).)*)\1(\s*,\s*context\s*=\s*([\"'])(?P<py_context>((?!\5).)*)\5)*(\s*,(\s*?.*?\n*?)*(,\s*([\"'])(?P<js_context>((?!\11).)*)\11)*)*\)")
|
||||||
|
words_pattern = re.compile(r"_{1,2}\([\"'`]{1,3}.*?[a-zA-Z]")
|
||||||
|
start_pattern = re.compile(r"_{1,2}\([f\"'`]{1,3}")
|
||||||
|
f_string_pattern = re.compile(r"_\(f[\"']")
|
||||||
|
starts_with_f_pattern = re.compile(r"_\(f")
|
||||||
|
|
||||||
|
# skip first argument
|
||||||
|
files = sys.argv[1:]
|
||||||
|
files_to_scan = [_file for _file in files if _file.endswith(('.py', '.js'))]
|
||||||
|
|
||||||
|
for _file in files_to_scan:
|
||||||
|
with open(_file, 'r') as f:
|
||||||
|
print(f'Checking: {_file}')
|
||||||
|
file_lines = f.readlines()
|
||||||
|
for line_number, line in enumerate(file_lines, 1):
|
||||||
|
if 'frappe-lint: disable-translate' in line:
|
||||||
|
continue
|
||||||
|
|
||||||
|
start_matches = start_pattern.search(line)
|
||||||
|
if start_matches:
|
||||||
|
starts_with_f = starts_with_f_pattern.search(line)
|
||||||
|
|
||||||
|
if starts_with_f:
|
||||||
|
has_f_string = f_string_pattern.search(line)
|
||||||
|
if has_f_string:
|
||||||
|
errors_encounter += 1
|
||||||
|
print(f'\nF-strings are not supported for translations at line number {line_number}\n{line.strip()[:100]}')
|
||||||
|
continue
|
||||||
|
else:
|
||||||
|
continue
|
||||||
|
|
||||||
|
match = pattern.search(line)
|
||||||
|
error_found = False
|
||||||
|
|
||||||
|
if not match and line.endswith((',\n', '[\n')):
|
||||||
|
# concat remaining text to validate multiline pattern
|
||||||
|
line = "".join(file_lines[line_number - 1:])
|
||||||
|
line = line[start_matches.start() + 1:]
|
||||||
|
match = pattern.match(line)
|
||||||
|
|
||||||
|
if not match:
|
||||||
|
error_found = True
|
||||||
|
print(f'\nTranslation syntax error at line number {line_number}\n{line.strip()[:100]}')
|
||||||
|
|
||||||
|
if not error_found and not words_pattern.search(line):
|
||||||
|
error_found = True
|
||||||
|
print(f'\nTranslation is useless because it has no words at line number {line_number}\n{line.strip()[:100]}')
|
||||||
|
|
||||||
|
if error_found:
|
||||||
|
errors_encounter += 1
|
||||||
|
|
||||||
|
if errors_encounter > 0:
|
||||||
|
print('\nVisit "https://frappeframework.com/docs/user/en/translations" to learn about valid translation strings.')
|
||||||
|
sys.exit(1)
|
||||||
|
else:
|
||||||
|
print('\nGood To Go!')
|
||||||
26
.github/workflows/backport.yml
vendored
Normal file
26
.github/workflows/backport.yml
vendored
Normal file
@@ -0,0 +1,26 @@
|
|||||||
|
name: Backport
|
||||||
|
on:
|
||||||
|
pull_request_target:
|
||||||
|
types:
|
||||||
|
- closed
|
||||||
|
- labeled
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
main:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
timeout-minutes: 60
|
||||||
|
steps:
|
||||||
|
- name: Checkout Actions
|
||||||
|
uses: actions/checkout@v2
|
||||||
|
with:
|
||||||
|
repository: "frappe/backport"
|
||||||
|
path: ./actions
|
||||||
|
ref: develop
|
||||||
|
- name: Install Actions
|
||||||
|
run: npm install --production --prefix ./actions
|
||||||
|
- name: Run backport
|
||||||
|
uses: ./actions/backport
|
||||||
|
with:
|
||||||
|
token: ${{secrets.RELEASE_TOKEN}}
|
||||||
|
labelsToAdd: "backport"
|
||||||
|
title: "{{originalTitle}}"
|
||||||
32
.github/workflows/initiate_release.yml
vendored
Normal file
32
.github/workflows/initiate_release.yml
vendored
Normal file
@@ -0,0 +1,32 @@
|
|||||||
|
# This workflow is agnostic to branches. Only maintain on develop branch.
|
||||||
|
# To add/remove versions just modify the matrix.
|
||||||
|
|
||||||
|
name: Create weekly release pull requests
|
||||||
|
on:
|
||||||
|
schedule:
|
||||||
|
# 9:30 UTC => 3 PM IST Tuesday
|
||||||
|
- cron: "30 9 * * 2"
|
||||||
|
workflow_dispatch:
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
stable-release:
|
||||||
|
name: Release
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
strategy:
|
||||||
|
fail-fast: false
|
||||||
|
matrix:
|
||||||
|
version: ["14", "15"]
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- uses: octokit/request-action@v2.x
|
||||||
|
with:
|
||||||
|
route: POST /repos/{owner}/{repo}/pulls
|
||||||
|
owner: frappe
|
||||||
|
repo: erpnext
|
||||||
|
title: |-
|
||||||
|
"chore: release v${{ matrix.version }}"
|
||||||
|
body: "Automated weekly release."
|
||||||
|
base: version-${{ matrix.version }}
|
||||||
|
head: version-${{ matrix.version }}-hotfix
|
||||||
|
env:
|
||||||
|
GITHUB_TOKEN: ${{ secrets.RELEASE_TOKEN }}
|
||||||
21
.github/workflows/lock.yml
vendored
Normal file
21
.github/workflows/lock.yml
vendored
Normal file
@@ -0,0 +1,21 @@
|
|||||||
|
name: 'Lock threads'
|
||||||
|
|
||||||
|
on:
|
||||||
|
schedule:
|
||||||
|
- cron: '0 0 * * *'
|
||||||
|
workflow_dispatch:
|
||||||
|
|
||||||
|
permissions:
|
||||||
|
issues: write
|
||||||
|
pull-requests: write
|
||||||
|
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
lock:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- uses: dessant/lock-threads@v5
|
||||||
|
with:
|
||||||
|
github-token: ${{ github.token }}
|
||||||
|
issue-inactive-days: 14
|
||||||
|
pr-inactive-days: 14
|
||||||
1
.github/workflows/patch.yml
vendored
1
.github/workflows/patch.yml
vendored
@@ -134,6 +134,7 @@ jobs:
|
|||||||
}
|
}
|
||||||
|
|
||||||
update_to_version 14
|
update_to_version 14
|
||||||
|
update_to_version 15
|
||||||
|
|
||||||
echo "Updating to latest version"
|
echo "Updating to latest version"
|
||||||
git -C "apps/frappe" checkout -q -f "${GITHUB_BASE_REF:-${GITHUB_REF##*/}}"
|
git -C "apps/frappe" checkout -q -f "${GITHUB_BASE_REF:-${GITHUB_REF##*/}}"
|
||||||
|
|||||||
8
.github/workflows/release.yml
vendored
8
.github/workflows/release.yml
vendored
@@ -2,23 +2,21 @@ name: Generate Semantic Release
|
|||||||
on:
|
on:
|
||||||
push:
|
push:
|
||||||
branches:
|
branches:
|
||||||
- version-15
|
- version-13
|
||||||
jobs:
|
jobs:
|
||||||
release:
|
release:
|
||||||
name: Release
|
name: Release
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout Entire Repository
|
- name: Checkout Entire Repository
|
||||||
uses: actions/checkout@v4
|
uses: actions/checkout@v2
|
||||||
with:
|
with:
|
||||||
fetch-depth: 0
|
fetch-depth: 0
|
||||||
persist-credentials: false
|
persist-credentials: false
|
||||||
|
|
||||||
- name: Setup Node.js
|
- name: Setup Node.js
|
||||||
uses: actions/setup-node@v2
|
uses: actions/setup-node@v2
|
||||||
with:
|
with:
|
||||||
node-version: 20
|
node-version: 18
|
||||||
|
|
||||||
- name: Setup dependencies
|
- name: Setup dependencies
|
||||||
run: |
|
run: |
|
||||||
npm install @semantic-release/git @semantic-release/exec --no-save
|
npm install @semantic-release/git @semantic-release/exec --no-save
|
||||||
|
|||||||
26
.github/workflows/server-tests-mariadb.yml
vendored
26
.github/workflows/server-tests-mariadb.yml
vendored
@@ -117,7 +117,7 @@ jobs:
|
|||||||
FRAPPE_BRANCH: ${{ github.event.inputs.branch }}
|
FRAPPE_BRANCH: ${{ github.event.inputs.branch }}
|
||||||
|
|
||||||
- name: Run Tests
|
- name: Run Tests
|
||||||
run: 'cd ~/frappe-bench/ && bench --site test_site run-parallel-tests --app erpnext --total-builds 4 --build-number ${{ matrix.container }}'
|
run: 'cd ~/frappe-bench/ && bench --site test_site run-parallel-tests --app erpnext --with-coverage --total-builds 4 --build-number ${{ matrix.container }}'
|
||||||
env:
|
env:
|
||||||
TYPE: server
|
TYPE: server
|
||||||
CI_BUILD_ID: ${{ github.run_id }}
|
CI_BUILD_ID: ${{ github.run_id }}
|
||||||
@@ -126,3 +126,27 @@ jobs:
|
|||||||
- name: Show bench output
|
- name: Show bench output
|
||||||
if: ${{ always() }}
|
if: ${{ always() }}
|
||||||
run: cat ~/frappe-bench/bench_start.log || true
|
run: cat ~/frappe-bench/bench_start.log || true
|
||||||
|
|
||||||
|
- name: Upload coverage data
|
||||||
|
uses: actions/upload-artifact@v3
|
||||||
|
with:
|
||||||
|
name: coverage-${{ matrix.container }}
|
||||||
|
path: /home/runner/frappe-bench/sites/coverage.xml
|
||||||
|
|
||||||
|
coverage:
|
||||||
|
name: Coverage Wrap Up
|
||||||
|
needs: test
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- name: Clone
|
||||||
|
uses: actions/checkout@v2
|
||||||
|
|
||||||
|
- name: Download artifacts
|
||||||
|
uses: actions/download-artifact@v3
|
||||||
|
|
||||||
|
- name: Upload coverage data
|
||||||
|
uses: codecov/codecov-action@v2
|
||||||
|
with:
|
||||||
|
name: MariaDB
|
||||||
|
fail_ci_if_error: true
|
||||||
|
verbose: true
|
||||||
|
|||||||
46
.mergify.yml
46
.mergify.yml
@@ -17,6 +17,7 @@ pull_request_rules:
|
|||||||
- base=version-12
|
- base=version-12
|
||||||
- base=version-14
|
- base=version-14
|
||||||
- base=version-15
|
- base=version-15
|
||||||
|
- base=version-16
|
||||||
actions:
|
actions:
|
||||||
close:
|
close:
|
||||||
comment:
|
comment:
|
||||||
@@ -24,16 +25,6 @@ pull_request_rules:
|
|||||||
@{{author}}, thanks for the contribution, but we do not accept pull requests on a stable branch. Please raise PR on an appropriate hotfix branch.
|
@{{author}}, thanks for the contribution, but we do not accept pull requests on a stable branch. Please raise PR on an appropriate hotfix branch.
|
||||||
https://github.com/frappe/erpnext/wiki/Pull-Request-Checklist#which-branch
|
https://github.com/frappe/erpnext/wiki/Pull-Request-Checklist#which-branch
|
||||||
|
|
||||||
- name: Auto-close PRs on pre-release branch
|
|
||||||
conditions:
|
|
||||||
- base=version-13-pre-release
|
|
||||||
actions:
|
|
||||||
close:
|
|
||||||
comment:
|
|
||||||
message: |
|
|
||||||
@{{author}}, pre-release branch is not maintained anymore. Releases are directly done by merging hotfix branch to stable branches.
|
|
||||||
|
|
||||||
|
|
||||||
- name: backport to develop
|
- name: backport to develop
|
||||||
conditions:
|
conditions:
|
||||||
- label="backport develop"
|
- label="backport develop"
|
||||||
@@ -54,13 +45,13 @@ pull_request_rules:
|
|||||||
assignees:
|
assignees:
|
||||||
- "{{ author }}"
|
- "{{ author }}"
|
||||||
|
|
||||||
- name: backport to version-14-pre-release
|
- name: backport to version-15-hotfix
|
||||||
conditions:
|
conditions:
|
||||||
- label="backport version-14-pre-release"
|
- label="backport version-15-hotfix"
|
||||||
actions:
|
actions:
|
||||||
backport:
|
backport:
|
||||||
branches:
|
branches:
|
||||||
- version-14-pre-release
|
- version-15-hotfix
|
||||||
assignees:
|
assignees:
|
||||||
- "{{ author }}"
|
- "{{ author }}"
|
||||||
|
|
||||||
@@ -74,35 +65,6 @@ pull_request_rules:
|
|||||||
assignees:
|
assignees:
|
||||||
- "{{ author }}"
|
- "{{ author }}"
|
||||||
|
|
||||||
- name: backport to version-13-pre-release
|
|
||||||
conditions:
|
|
||||||
- label="backport version-13-pre-release"
|
|
||||||
actions:
|
|
||||||
backport:
|
|
||||||
branches:
|
|
||||||
- version-13-pre-release
|
|
||||||
assignees:
|
|
||||||
- "{{ author }}"
|
|
||||||
|
|
||||||
- name: backport to version-12-hotfix
|
|
||||||
conditions:
|
|
||||||
- label="backport version-12-hotfix"
|
|
||||||
actions:
|
|
||||||
backport:
|
|
||||||
branches:
|
|
||||||
- version-12-hotfix
|
|
||||||
assignees:
|
|
||||||
- "{{ author }}"
|
|
||||||
|
|
||||||
- name: backport to version-12-pre-release
|
|
||||||
conditions:
|
|
||||||
- label="backport version-12-pre-release"
|
|
||||||
actions:
|
|
||||||
backport:
|
|
||||||
branches:
|
|
||||||
- version-12-pre-release
|
|
||||||
assignees:
|
|
||||||
- "{{ author }}"
|
|
||||||
|
|
||||||
- name: Automatic merge on CI success and review
|
- name: Automatic merge on CI success and review
|
||||||
conditions:
|
conditions:
|
||||||
|
|||||||
@@ -20,23 +20,6 @@ repos:
|
|||||||
- id: check-yaml
|
- id: check-yaml
|
||||||
- id: debug-statements
|
- id: debug-statements
|
||||||
|
|
||||||
- repo: https://github.com/pre-commit/mirrors-prettier
|
|
||||||
rev: v2.7.1
|
|
||||||
hooks:
|
|
||||||
- id: prettier
|
|
||||||
types_or: [javascript, vue, scss]
|
|
||||||
# Ignore any files that might contain jinja / bundles
|
|
||||||
exclude: |
|
|
||||||
(?x)^(
|
|
||||||
erpnext/public/dist/.*|
|
|
||||||
cypress/.*|
|
|
||||||
.*node_modules.*|
|
|
||||||
.*boilerplate.*|
|
|
||||||
erpnext/public/js/controllers/.*|
|
|
||||||
erpnext/templates/pages/order.js|
|
|
||||||
erpnext/templates/includes/.*
|
|
||||||
)$
|
|
||||||
|
|
||||||
- repo: https://github.com/pre-commit/mirrors-eslint
|
- repo: https://github.com/pre-commit/mirrors-eslint
|
||||||
rev: v8.44.0
|
rev: v8.44.0
|
||||||
hooks:
|
hooks:
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
{
|
{
|
||||||
"branches": ["version-15"],
|
"branches": ["version-13"],
|
||||||
"plugins": [
|
"plugins": [
|
||||||
"@semantic-release/commit-analyzer", {
|
"@semantic-release/commit-analyzer", {
|
||||||
"preset": "angular",
|
"preset": "angular",
|
||||||
@@ -21,4 +21,4 @@
|
|||||||
],
|
],
|
||||||
"@semantic-release/github"
|
"@semantic-release/github"
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
@@ -73,8 +73,6 @@ New passwords will be created for the ERPNext "Administrator" user, the MariaDB
|
|||||||
1. [Issue Guidelines](https://github.com/frappe/erpnext/wiki/Issue-Guidelines)
|
1. [Issue Guidelines](https://github.com/frappe/erpnext/wiki/Issue-Guidelines)
|
||||||
1. [Report Security Vulnerabilities](https://erpnext.com/security)
|
1. [Report Security Vulnerabilities](https://erpnext.com/security)
|
||||||
1. [Pull Request Requirements](https://github.com/frappe/erpnext/wiki/Contribution-Guidelines)
|
1. [Pull Request Requirements](https://github.com/frappe/erpnext/wiki/Contribution-Guidelines)
|
||||||
1. [Translations](https://translate.erpnext.com)
|
|
||||||
|
|
||||||
|
|
||||||
## License
|
## License
|
||||||
|
|
||||||
|
|||||||
@@ -1,13 +1,25 @@
|
|||||||
module.exports = {
|
module.exports = {
|
||||||
parserPreset: "conventional-changelog-conventionalcommits",
|
parserPreset: 'conventional-changelog-conventionalcommits',
|
||||||
rules: {
|
rules: {
|
||||||
"subject-empty": [2, "never"],
|
'subject-empty': [2, 'never'],
|
||||||
"type-case": [2, "always", "lower-case"],
|
'type-case': [2, 'always', 'lower-case'],
|
||||||
"type-empty": [2, "never"],
|
'type-empty': [2, 'never'],
|
||||||
"type-enum": [
|
'type-enum': [
|
||||||
2,
|
2,
|
||||||
"always",
|
'always',
|
||||||
["build", "chore", "ci", "docs", "feat", "fix", "perf", "refactor", "revert", "style", "test"],
|
[
|
||||||
|
'build',
|
||||||
|
'chore',
|
||||||
|
'ci',
|
||||||
|
'docs',
|
||||||
|
'feat',
|
||||||
|
'fix',
|
||||||
|
'perf',
|
||||||
|
'refactor',
|
||||||
|
'revert',
|
||||||
|
'style',
|
||||||
|
'test',
|
||||||
|
],
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -3,7 +3,7 @@ import inspect
|
|||||||
|
|
||||||
import frappe
|
import frappe
|
||||||
|
|
||||||
__version__ = "15.16.2"
|
__version__ = "16.0.0-dev"
|
||||||
|
|
||||||
|
|
||||||
def get_default_company(user=None):
|
def get_default_company(user=None):
|
||||||
@@ -13,7 +13,7 @@ def get_default_company(user=None):
|
|||||||
if not user:
|
if not user:
|
||||||
user = frappe.session.user
|
user = frappe.session.user
|
||||||
|
|
||||||
companies = get_user_default_as_list("company", user)
|
companies = get_user_default_as_list(user, "company")
|
||||||
if companies:
|
if companies:
|
||||||
default_company = companies[0]
|
default_company = companies[0]
|
||||||
else:
|
else:
|
||||||
@@ -36,7 +36,7 @@ def get_default_cost_center(company):
|
|||||||
|
|
||||||
if not frappe.flags.company_cost_center:
|
if not frappe.flags.company_cost_center:
|
||||||
frappe.flags.company_cost_center = {}
|
frappe.flags.company_cost_center = {}
|
||||||
if not company in frappe.flags.company_cost_center:
|
if company not in frappe.flags.company_cost_center:
|
||||||
frappe.flags.company_cost_center[company] = frappe.get_cached_value(
|
frappe.flags.company_cost_center[company] = frappe.get_cached_value(
|
||||||
"Company", company, "cost_center"
|
"Company", company, "cost_center"
|
||||||
)
|
)
|
||||||
@@ -47,7 +47,7 @@ def get_company_currency(company):
|
|||||||
"""Returns the default company currency"""
|
"""Returns the default company currency"""
|
||||||
if not frappe.flags.company_currency:
|
if not frappe.flags.company_currency:
|
||||||
frappe.flags.company_currency = {}
|
frappe.flags.company_currency = {}
|
||||||
if not company in frappe.flags.company_currency:
|
if company not in frappe.flags.company_currency:
|
||||||
frappe.flags.company_currency[company] = frappe.db.get_value(
|
frappe.flags.company_currency[company] = frappe.db.get_value(
|
||||||
"Company", company, "default_currency", cache=True
|
"Company", company, "default_currency", cache=True
|
||||||
)
|
)
|
||||||
@@ -81,7 +81,7 @@ def is_perpetual_inventory_enabled(company):
|
|||||||
if not hasattr(frappe.local, "enable_perpetual_inventory"):
|
if not hasattr(frappe.local, "enable_perpetual_inventory"):
|
||||||
frappe.local.enable_perpetual_inventory = {}
|
frappe.local.enable_perpetual_inventory = {}
|
||||||
|
|
||||||
if not company in frappe.local.enable_perpetual_inventory:
|
if company not in frappe.local.enable_perpetual_inventory:
|
||||||
frappe.local.enable_perpetual_inventory[company] = (
|
frappe.local.enable_perpetual_inventory[company] = (
|
||||||
frappe.get_cached_value("Company", company, "enable_perpetual_inventory") or 0
|
frappe.get_cached_value("Company", company, "enable_perpetual_inventory") or 0
|
||||||
)
|
)
|
||||||
@@ -96,7 +96,7 @@ def get_default_finance_book(company=None):
|
|||||||
if not hasattr(frappe.local, "default_finance_book"):
|
if not hasattr(frappe.local, "default_finance_book"):
|
||||||
frappe.local.default_finance_book = {}
|
frappe.local.default_finance_book = {}
|
||||||
|
|
||||||
if not company in frappe.local.default_finance_book:
|
if company not in frappe.local.default_finance_book:
|
||||||
frappe.local.default_finance_book[company] = frappe.get_cached_value(
|
frappe.local.default_finance_book[company] = frappe.get_cached_value(
|
||||||
"Company", company, "default_finance_book"
|
"Company", company, "default_finance_book"
|
||||||
)
|
)
|
||||||
@@ -108,7 +108,7 @@ def get_party_account_type(party_type):
|
|||||||
if not hasattr(frappe.local, "party_account_types"):
|
if not hasattr(frappe.local, "party_account_types"):
|
||||||
frappe.local.party_account_types = {}
|
frappe.local.party_account_types = {}
|
||||||
|
|
||||||
if not party_type in frappe.local.party_account_types:
|
if party_type not in frappe.local.party_account_types:
|
||||||
frappe.local.party_account_types[party_type] = (
|
frappe.local.party_account_types[party_type] = (
|
||||||
frappe.db.get_value("Party Type", party_type, "account_type") or ""
|
frappe.db.get_value("Party Type", party_type, "account_type") or ""
|
||||||
)
|
)
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
frappe.provide("frappe.dashboards.chart_sources");
|
frappe.provide('frappe.dashboards.chart_sources');
|
||||||
|
|
||||||
frappe.dashboards.chart_sources["Account Balance Timeline"] = {
|
frappe.dashboards.chart_sources["Account Balance Timeline"] = {
|
||||||
method: "erpnext.accounts.dashboard_chart_source.account_balance_timeline.account_balance_timeline.get",
|
method: "erpnext.accounts.dashboard_chart_source.account_balance_timeline.account_balance_timeline.get",
|
||||||
@@ -9,14 +9,14 @@ frappe.dashboards.chart_sources["Account Balance Timeline"] = {
|
|||||||
fieldtype: "Link",
|
fieldtype: "Link",
|
||||||
options: "Company",
|
options: "Company",
|
||||||
default: frappe.defaults.get_user_default("Company"),
|
default: frappe.defaults.get_user_default("Company"),
|
||||||
reqd: 1,
|
reqd: 1
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
fieldname: "account",
|
fieldname: "account",
|
||||||
label: __("Account"),
|
label: __("Account"),
|
||||||
fieldtype: "Link",
|
fieldtype: "Link",
|
||||||
options: "Account",
|
options: "Account",
|
||||||
reqd: 1,
|
reqd: 1
|
||||||
},
|
},
|
||||||
],
|
]
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -232,7 +232,7 @@ def calculate_monthly_amount(
|
|||||||
if amount + already_booked_amount_in_account_currency > item.net_amount:
|
if amount + already_booked_amount_in_account_currency > item.net_amount:
|
||||||
amount = item.net_amount - already_booked_amount_in_account_currency
|
amount = item.net_amount - already_booked_amount_in_account_currency
|
||||||
|
|
||||||
if not (get_first_day(start_date) == start_date and get_last_day(end_date) == end_date):
|
if get_first_day(start_date) != start_date or get_last_day(end_date) != end_date:
|
||||||
partial_month = flt(date_diff(end_date, start_date)) / flt(
|
partial_month = flt(date_diff(end_date, start_date)) / flt(
|
||||||
date_diff(get_last_day(end_date), get_first_day(start_date))
|
date_diff(get_last_day(end_date), get_first_day(start_date))
|
||||||
)
|
)
|
||||||
@@ -358,9 +358,11 @@ def book_deferred_income_or_expense(doc, deferred_process, posting_date=None):
|
|||||||
|
|
||||||
account_currency = get_account_currency(item.expense_account or item.income_account)
|
account_currency = get_account_currency(item.expense_account or item.income_account)
|
||||||
if doc.doctype == "Sales Invoice":
|
if doc.doctype == "Sales Invoice":
|
||||||
|
against_type = "Customer"
|
||||||
against, project = doc.customer, doc.project
|
against, project = doc.customer, doc.project
|
||||||
credit_account, debit_account = item.income_account, item.deferred_revenue_account
|
credit_account, debit_account = item.income_account, item.deferred_revenue_account
|
||||||
else:
|
else:
|
||||||
|
against_type = "Supplier"
|
||||||
against, project = doc.supplier, item.project
|
against, project = doc.supplier, item.project
|
||||||
credit_account, debit_account = item.deferred_expense_account, item.expense_account
|
credit_account, debit_account = item.deferred_expense_account, item.expense_account
|
||||||
|
|
||||||
@@ -413,6 +415,7 @@ def book_deferred_income_or_expense(doc, deferred_process, posting_date=None):
|
|||||||
doc,
|
doc,
|
||||||
credit_account,
|
credit_account,
|
||||||
debit_account,
|
debit_account,
|
||||||
|
against_type,
|
||||||
against,
|
against,
|
||||||
amount,
|
amount,
|
||||||
base_amount,
|
base_amount,
|
||||||
@@ -494,6 +497,7 @@ def make_gl_entries(
|
|||||||
doc,
|
doc,
|
||||||
credit_account,
|
credit_account,
|
||||||
debit_account,
|
debit_account,
|
||||||
|
against_type,
|
||||||
against,
|
against,
|
||||||
amount,
|
amount,
|
||||||
base_amount,
|
base_amount,
|
||||||
@@ -515,7 +519,9 @@ def make_gl_entries(
|
|||||||
doc.get_gl_dict(
|
doc.get_gl_dict(
|
||||||
{
|
{
|
||||||
"account": credit_account,
|
"account": credit_account,
|
||||||
|
"against_type": against_type,
|
||||||
"against": against,
|
"against": against,
|
||||||
|
"against_link": against,
|
||||||
"credit": base_amount,
|
"credit": base_amount,
|
||||||
"credit_in_account_currency": amount,
|
"credit_in_account_currency": amount,
|
||||||
"cost_center": cost_center,
|
"cost_center": cost_center,
|
||||||
@@ -534,7 +540,9 @@ def make_gl_entries(
|
|||||||
doc.get_gl_dict(
|
doc.get_gl_dict(
|
||||||
{
|
{
|
||||||
"account": debit_account,
|
"account": debit_account,
|
||||||
|
"against_type": against_type,
|
||||||
"against": against,
|
"against": against,
|
||||||
|
"against_link": against,
|
||||||
"debit": base_amount,
|
"debit": base_amount,
|
||||||
"debit_in_account_currency": amount,
|
"debit_in_account_currency": amount,
|
||||||
"cost_center": cost_center,
|
"cost_center": cost_center,
|
||||||
|
|||||||
@@ -26,14 +26,19 @@ frappe.ui.form.on("Account", {
|
|||||||
frm.toggle_enable(["is_group", "company"], false);
|
frm.toggle_enable(["is_group", "company"], false);
|
||||||
|
|
||||||
if (cint(frm.doc.is_group) == 0) {
|
if (cint(frm.doc.is_group) == 0) {
|
||||||
frm.toggle_display("freeze_account", frm.doc.__onload && frm.doc.__onload.can_freeze_account);
|
frm.toggle_display(
|
||||||
|
"freeze_account",
|
||||||
|
frm.doc.__onload && frm.doc.__onload.can_freeze_account
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
// read-only for root accounts
|
// read-only for root accounts
|
||||||
if (!frm.is_new()) {
|
if (!frm.is_new()) {
|
||||||
if (!frm.doc.parent_account) {
|
if (!frm.doc.parent_account) {
|
||||||
frm.set_read_only();
|
frm.set_read_only();
|
||||||
frm.set_intro(__("This is a root account and cannot be edited."));
|
frm.set_intro(
|
||||||
|
__("This is a root account and cannot be edited.")
|
||||||
|
);
|
||||||
} else {
|
} else {
|
||||||
// credit days and type if customer or supplier
|
// credit days and type if customer or supplier
|
||||||
frm.set_intro(null);
|
frm.set_intro(null);
|
||||||
@@ -75,33 +80,27 @@ frappe.ui.form.on("Account", {
|
|||||||
);
|
);
|
||||||
|
|
||||||
if (frm.doc.is_group == 1) {
|
if (frm.doc.is_group == 1) {
|
||||||
frm.add_custom_button(
|
frm.add_custom_button(__('Convert to Non-Group'), function () {
|
||||||
__("Convert to Non-Group"),
|
return frappe.call({
|
||||||
function () {
|
doc: frm.doc,
|
||||||
return frappe.call({
|
method: 'convert_group_to_ledger',
|
||||||
doc: frm.doc,
|
callback: function() {
|
||||||
method: "convert_group_to_ledger",
|
frm.refresh();
|
||||||
callback: function () {
|
}
|
||||||
frm.refresh();
|
});
|
||||||
},
|
}, __('Actions'));
|
||||||
});
|
|
||||||
},
|
} else if (cint(frm.doc.is_group) == 0
|
||||||
__("Actions")
|
&& frappe.boot.user.can_read.indexOf("GL Entry") !== -1) {
|
||||||
);
|
frm.add_custom_button(__('General Ledger'), function () {
|
||||||
} else if (cint(frm.doc.is_group) == 0 && frappe.boot.user.can_read.indexOf("GL Entry") !== -1) {
|
frappe.route_options = {
|
||||||
frm.add_custom_button(
|
"account": frm.doc.name,
|
||||||
__("General Ledger"),
|
"from_date": erpnext.utils.get_fiscal_year(frappe.datetime.get_today(), true)[1],
|
||||||
function () {
|
"to_date": erpnext.utils.get_fiscal_year(frappe.datetime.get_today(), true)[2],
|
||||||
frappe.route_options = {
|
"company": frm.doc.company
|
||||||
account: frm.doc.name,
|
};
|
||||||
from_date: erpnext.utils.get_fiscal_year(frappe.datetime.get_today(), true)[1],
|
frappe.set_route("query-report", "General Ledger");
|
||||||
to_date: erpnext.utils.get_fiscal_year(frappe.datetime.get_today(), true)[2],
|
}, __('View'));
|
||||||
company: frm.doc.company,
|
|
||||||
};
|
|
||||||
frappe.set_route("query-report", "General Ledger");
|
|
||||||
},
|
|
||||||
__("View")
|
|
||||||
);
|
|
||||||
|
|
||||||
frm.add_custom_button(
|
frm.add_custom_button(
|
||||||
__("Convert to Group"),
|
__("Convert to Group"),
|
||||||
@@ -194,8 +193,14 @@ frappe.ui.form.on("Account", {
|
|||||||
if (r.message) {
|
if (r.message) {
|
||||||
frappe.set_route("Form", "Account", r.message);
|
frappe.set_route("Form", "Account", r.message);
|
||||||
} else {
|
} else {
|
||||||
frm.set_value("account_number", data.account_number);
|
frm.set_value(
|
||||||
frm.set_value("account_name", data.account_name);
|
"account_number",
|
||||||
|
data.account_number
|
||||||
|
);
|
||||||
|
frm.set_value(
|
||||||
|
"account_name",
|
||||||
|
data.account_name
|
||||||
|
);
|
||||||
}
|
}
|
||||||
d.hide();
|
d.hide();
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -91,8 +91,8 @@ class Account(NestedSet):
|
|||||||
super(Account, self).on_update()
|
super(Account, self).on_update()
|
||||||
|
|
||||||
def onload(self):
|
def onload(self):
|
||||||
frozen_accounts_modifier = frappe.db.get_value(
|
frozen_accounts_modifier = frappe.db.get_single_value(
|
||||||
"Accounts Settings", "Accounts Settings", "frozen_accounts_modifier"
|
"Accounts Settings", "frozen_accounts_modifier"
|
||||||
)
|
)
|
||||||
if not frozen_accounts_modifier or frozen_accounts_modifier in frappe.get_roles():
|
if not frozen_accounts_modifier or frozen_accounts_modifier in frappe.get_roles():
|
||||||
self.set_onload("can_freeze_account", True)
|
self.set_onload("can_freeze_account", True)
|
||||||
@@ -118,7 +118,6 @@ class Account(NestedSet):
|
|||||||
self.validate_balance_must_be_debit_or_credit()
|
self.validate_balance_must_be_debit_or_credit()
|
||||||
self.validate_account_currency()
|
self.validate_account_currency()
|
||||||
self.validate_root_company_and_sync_account_to_children()
|
self.validate_root_company_and_sync_account_to_children()
|
||||||
self.validate_receivable_payable_account_type()
|
|
||||||
|
|
||||||
def validate_parent_child_account_type(self):
|
def validate_parent_child_account_type(self):
|
||||||
if self.parent_account:
|
if self.parent_account:
|
||||||
@@ -189,24 +188,6 @@ class Account(NestedSet):
|
|||||||
"Balance Sheet" if self.root_type in ("Asset", "Liability", "Equity") else "Profit and Loss"
|
"Balance Sheet" if self.root_type in ("Asset", "Liability", "Equity") else "Profit and Loss"
|
||||||
)
|
)
|
||||||
|
|
||||||
def validate_receivable_payable_account_type(self):
|
|
||||||
doc_before_save = self.get_doc_before_save()
|
|
||||||
receivable_payable_types = ["Receivable", "Payable"]
|
|
||||||
if (
|
|
||||||
doc_before_save
|
|
||||||
and doc_before_save.account_type in receivable_payable_types
|
|
||||||
and doc_before_save.account_type != self.account_type
|
|
||||||
):
|
|
||||||
# check for ledger entries
|
|
||||||
if frappe.db.get_all("GL Entry", filters={"account": self.name, "is_cancelled": 0}, limit=1):
|
|
||||||
msg = _(
|
|
||||||
"There are ledger entries against this account. Changing {0} to non-{1} in live system will cause incorrect output in 'Accounts {2}' report"
|
|
||||||
).format(
|
|
||||||
frappe.bold("Account Type"), doc_before_save.account_type, doc_before_save.account_type
|
|
||||||
)
|
|
||||||
frappe.msgprint(msg)
|
|
||||||
self.add_comment("Comment", msg)
|
|
||||||
|
|
||||||
def validate_root_details(self):
|
def validate_root_details(self):
|
||||||
doc_before_save = self.get_doc_before_save()
|
doc_before_save = self.get_doc_before_save()
|
||||||
|
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
frappe.provide("frappe.treeview_settings");
|
frappe.provide("frappe.treeview_settings")
|
||||||
|
|
||||||
frappe.treeview_settings["Account"] = {
|
frappe.treeview_settings["Account"] = {
|
||||||
breadcrumb: "Accounts",
|
breadcrumb: "Accounts",
|
||||||
@@ -7,12 +7,12 @@ frappe.treeview_settings["Account"] = {
|
|||||||
filters: [
|
filters: [
|
||||||
{
|
{
|
||||||
fieldname: "company",
|
fieldname: "company",
|
||||||
fieldtype: "Select",
|
fieldtype:"Select",
|
||||||
options: erpnext.utils.get_tree_options("company"),
|
options: erpnext.utils.get_tree_options("company"),
|
||||||
label: __("Company"),
|
label: __("Company"),
|
||||||
default: erpnext.utils.get_tree_default("company"),
|
default: erpnext.utils.get_tree_default("company"),
|
||||||
on_change: function () {
|
on_change: function() {
|
||||||
var me = frappe.treeview_settings["Account"].treeview;
|
var me = frappe.treeview_settings['Account'].treeview;
|
||||||
var company = me.page.fields_dict.company.get_value();
|
var company = me.page.fields_dict.company.get_value();
|
||||||
if (!company) {
|
if (!company) {
|
||||||
frappe.throw(__("Please set a Company"));
|
frappe.throw(__("Please set a Company"));
|
||||||
@@ -22,36 +22,30 @@ frappe.treeview_settings["Account"] = {
|
|||||||
args: {
|
args: {
|
||||||
company: company,
|
company: company,
|
||||||
},
|
},
|
||||||
callback: function (r) {
|
callback: function(r) {
|
||||||
if (r.message) {
|
if(r.message) {
|
||||||
let root_company = r.message.length ? r.message[0] : "";
|
let root_company = r.message.length ? r.message[0] : "";
|
||||||
me.page.fields_dict.root_company.set_value(root_company);
|
me.page.fields_dict.root_company.set_value(root_company);
|
||||||
|
|
||||||
frappe.db.get_value(
|
frappe.db.get_value("Company", {"name": company}, "allow_account_creation_against_child_company", (r) => {
|
||||||
"Company",
|
frappe.flags.ignore_root_company_validation = r.allow_account_creation_against_child_company;
|
||||||
{ name: company },
|
});
|
||||||
"allow_account_creation_against_child_company",
|
|
||||||
(r) => {
|
|
||||||
frappe.flags.ignore_root_company_validation =
|
|
||||||
r.allow_account_creation_against_child_company;
|
|
||||||
}
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
},
|
}
|
||||||
});
|
});
|
||||||
},
|
}
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
fieldname: "root_company",
|
fieldname: "root_company",
|
||||||
fieldtype: "Data",
|
fieldtype:"Data",
|
||||||
label: __("Root Company"),
|
label: __("Root Company"),
|
||||||
hidden: true,
|
hidden: true,
|
||||||
disable_onchange: true,
|
disable_onchange: true
|
||||||
},
|
}
|
||||||
],
|
],
|
||||||
root_label: "Accounts",
|
root_label: "Accounts",
|
||||||
get_tree_nodes: "erpnext.accounts.utils.get_children",
|
get_tree_nodes: 'erpnext.accounts.utils.get_children',
|
||||||
on_get_node: function (nodes, deep = false) {
|
on_get_node: function(nodes, deep=false) {
|
||||||
if (frappe.boot.user.can_read.indexOf("GL Entry") == -1) return;
|
if (frappe.boot.user.can_read.indexOf("GL Entry") == -1) return;
|
||||||
|
|
||||||
let accounts = [];
|
let accounts = [];
|
||||||
@@ -63,231 +57,151 @@ frappe.treeview_settings["Account"] = {
|
|||||||
}
|
}
|
||||||
|
|
||||||
frappe.db.get_single_value("Accounts Settings", "show_balance_in_coa").then((value) => {
|
frappe.db.get_single_value("Accounts Settings", "show_balance_in_coa").then((value) => {
|
||||||
if (value) {
|
if(value) {
|
||||||
|
|
||||||
const get_balances = frappe.call({
|
const get_balances = frappe.call({
|
||||||
method: "erpnext.accounts.utils.get_account_balances",
|
method: 'erpnext.accounts.utils.get_account_balances',
|
||||||
args: {
|
args: {
|
||||||
accounts: accounts,
|
accounts: accounts,
|
||||||
company: cur_tree.args.company,
|
company: cur_tree.args.company
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
get_balances.then((r) => {
|
get_balances.then(r => {
|
||||||
if (!r.message || r.message.length == 0) return;
|
if (!r.message || r.message.length == 0) return;
|
||||||
|
|
||||||
for (let account of r.message) {
|
for (let account of r.message) {
|
||||||
|
|
||||||
const node = cur_tree.nodes && cur_tree.nodes[account.value];
|
const node = cur_tree.nodes && cur_tree.nodes[account.value];
|
||||||
if (!node || node.is_root) continue;
|
if (!node || node.is_root) continue;
|
||||||
|
|
||||||
// show Dr if positive since balance is calculated as debit - credit else show Cr
|
// show Dr if positive since balance is calculated as debit - credit else show Cr
|
||||||
const balance = account.balance_in_account_currency || account.balance;
|
const balance = account.balance_in_account_currency || account.balance;
|
||||||
const dr_or_cr = balance > 0 ? "Dr" : "Cr";
|
const dr_or_cr = balance > 0 ? __("Dr"): __("Cr");
|
||||||
const format = (value, currency) => format_currency(Math.abs(value), currency);
|
const format = (value, currency) => format_currency(Math.abs(value), currency);
|
||||||
|
|
||||||
if (account.balance !== undefined) {
|
if (account.balance!==undefined) {
|
||||||
node.parent && node.parent.find(".balance-area").remove();
|
node.parent && node.parent.find('.balance-area').remove();
|
||||||
$(
|
$('<span class="balance-area pull-right">'
|
||||||
'<span class="balance-area pull-right">' +
|
+ (account.balance_in_account_currency ?
|
||||||
(account.balance_in_account_currency
|
(format(account.balance_in_account_currency, account.account_currency) + " / ") : "")
|
||||||
? format(
|
+ format(account.balance, account.company_currency)
|
||||||
account.balance_in_account_currency,
|
+ " " + dr_or_cr
|
||||||
account.account_currency
|
+ '</span>').insertBefore(node.$ul);
|
||||||
) + " / "
|
|
||||||
: "") +
|
|
||||||
format(account.balance, account.company_currency) +
|
|
||||||
" " +
|
|
||||||
dr_or_cr +
|
|
||||||
"</span>"
|
|
||||||
).insertBefore(node.$ul);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
add_tree_node: "erpnext.accounts.utils.add_ac",
|
add_tree_node: 'erpnext.accounts.utils.add_ac',
|
||||||
menu_items: [
|
menu_items:[
|
||||||
{
|
{
|
||||||
label: __("New Company"),
|
label: __('New Company'),
|
||||||
action: function () {
|
action: function() { frappe.new_doc("Company", true) },
|
||||||
frappe.new_doc("Company", true);
|
condition: 'frappe.boot.user.can_create.indexOf("Company") !== -1'
|
||||||
},
|
}
|
||||||
condition: 'frappe.boot.user.can_create.indexOf("Company") !== -1',
|
|
||||||
},
|
|
||||||
],
|
],
|
||||||
fields: [
|
fields: [
|
||||||
{
|
{fieldtype:'Data', fieldname:'account_name', label:__('New Account Name'), reqd:true,
|
||||||
fieldtype: "Data",
|
description: __("Name of new Account. Note: Please don't create accounts for Customers and Suppliers")},
|
||||||
fieldname: "account_name",
|
{fieldtype:'Data', fieldname:'account_number', label:__('Account Number'),
|
||||||
label: __("New Account Name"),
|
description: __("Number of new Account, it will be included in the account name as a prefix")},
|
||||||
reqd: true,
|
{fieldtype:'Check', fieldname:'is_group', label:__('Is Group'),
|
||||||
description: __(
|
description: __('Further accounts can be made under Groups, but entries can be made against non-Groups')},
|
||||||
"Name of new Account. Note: Please don't create accounts for Customers and Suppliers"
|
{fieldtype:'Select', fieldname:'root_type', label:__('Root Type'),
|
||||||
),
|
options: ['Asset', 'Liability', 'Equity', 'Income', 'Expense'].join('\n'),
|
||||||
},
|
depends_on: 'eval:doc.is_group && !doc.parent_account'},
|
||||||
{
|
{fieldtype:'Select', fieldname:'account_type', label:__('Account Type'),
|
||||||
fieldtype: "Data",
|
options: frappe.get_meta("Account").fields.filter(d => d.fieldname=='account_type')[0].options,
|
||||||
fieldname: "account_number",
|
description: __("Optional. This setting will be used to filter in various transactions.")
|
||||||
label: __("Account Number"),
|
|
||||||
description: __("Number of new Account, it will be included in the account name as a prefix"),
|
|
||||||
},
|
|
||||||
{
|
|
||||||
fieldtype: "Check",
|
|
||||||
fieldname: "is_group",
|
|
||||||
label: __("Is Group"),
|
|
||||||
description: __(
|
|
||||||
"Further accounts can be made under Groups, but entries can be made against non-Groups"
|
|
||||||
),
|
|
||||||
},
|
|
||||||
{
|
|
||||||
fieldtype: "Select",
|
|
||||||
fieldname: "root_type",
|
|
||||||
label: __("Root Type"),
|
|
||||||
options: ["Asset", "Liability", "Equity", "Income", "Expense"].join("\n"),
|
|
||||||
depends_on: "eval:doc.is_group && !doc.parent_account",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
fieldtype: "Select",
|
|
||||||
fieldname: "account_type",
|
|
||||||
label: __("Account Type"),
|
|
||||||
options: frappe.get_meta("Account").fields.filter((d) => d.fieldname == "account_type")[0]
|
|
||||||
.options,
|
|
||||||
description: __("Optional. This setting will be used to filter in various transactions."),
|
|
||||||
},
|
|
||||||
{
|
|
||||||
fieldtype: "Float",
|
|
||||||
fieldname: "tax_rate",
|
|
||||||
label: __("Tax Rate"),
|
|
||||||
depends_on: 'eval:doc.is_group==0&&doc.account_type=="Tax"',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
fieldtype: "Link",
|
|
||||||
fieldname: "account_currency",
|
|
||||||
label: __("Currency"),
|
|
||||||
options: "Currency",
|
|
||||||
description: __("Optional. Sets company's default currency, if not specified."),
|
|
||||||
},
|
},
|
||||||
|
{fieldtype:'Float', fieldname:'tax_rate', label:__('Tax Rate'),
|
||||||
|
depends_on: 'eval:doc.is_group==0&&doc.account_type=="Tax"'},
|
||||||
|
{fieldtype:'Link', fieldname:'account_currency', label:__('Currency'), options:"Currency",
|
||||||
|
description: __("Optional. Sets company's default currency, if not specified.")}
|
||||||
],
|
],
|
||||||
ignore_fields: ["parent_account"],
|
ignore_fields:["parent_account"],
|
||||||
onload: function (treeview) {
|
onload: function(treeview) {
|
||||||
frappe.treeview_settings["Account"].treeview = {};
|
frappe.treeview_settings['Account'].treeview = {};
|
||||||
$.extend(frappe.treeview_settings["Account"].treeview, treeview);
|
$.extend(frappe.treeview_settings['Account'].treeview, treeview);
|
||||||
function get_company() {
|
function get_company() {
|
||||||
return treeview.page.fields_dict.company.get_value();
|
return treeview.page.fields_dict.company.get_value();
|
||||||
}
|
}
|
||||||
|
|
||||||
// tools
|
// tools
|
||||||
treeview.page.add_inner_button(
|
treeview.page.add_inner_button(__("Chart of Cost Centers"), function() {
|
||||||
__("Chart of Cost Centers"),
|
frappe.set_route('Tree', 'Cost Center', {company: get_company()});
|
||||||
function () {
|
}, __('View'));
|
||||||
frappe.set_route("Tree", "Cost Center", { company: get_company() });
|
|
||||||
},
|
|
||||||
__("View")
|
|
||||||
);
|
|
||||||
|
|
||||||
treeview.page.add_inner_button(
|
treeview.page.add_inner_button(__("Opening Invoice Creation Tool"), function() {
|
||||||
__("Opening Invoice Creation Tool"),
|
frappe.set_route('Form', 'Opening Invoice Creation Tool', {company: get_company()});
|
||||||
function () {
|
}, __('View'));
|
||||||
frappe.set_route("Form", "Opening Invoice Creation Tool", { company: get_company() });
|
|
||||||
},
|
|
||||||
__("View")
|
|
||||||
);
|
|
||||||
|
|
||||||
treeview.page.add_inner_button(
|
treeview.page.add_inner_button(__("Period Closing Voucher"), function() {
|
||||||
__("Period Closing Voucher"),
|
frappe.set_route('List', 'Period Closing Voucher', {company: get_company()});
|
||||||
function () {
|
}, __('View'));
|
||||||
frappe.set_route("List", "Period Closing Voucher", { company: get_company() });
|
|
||||||
},
|
|
||||||
__("View")
|
|
||||||
);
|
|
||||||
|
|
||||||
treeview.page.add_inner_button(
|
|
||||||
__("Journal Entry"),
|
treeview.page.add_inner_button(__("Journal Entry"), function() {
|
||||||
function () {
|
frappe.new_doc('Journal Entry', {company: get_company()});
|
||||||
frappe.new_doc("Journal Entry", { company: get_company() });
|
}, __('Create'));
|
||||||
},
|
treeview.page.add_inner_button(__("Company"), function() {
|
||||||
__("Create")
|
frappe.new_doc('Company');
|
||||||
);
|
}, __('Create'));
|
||||||
treeview.page.add_inner_button(
|
|
||||||
__("Company"),
|
|
||||||
function () {
|
|
||||||
frappe.new_doc("Company");
|
|
||||||
},
|
|
||||||
__("Create")
|
|
||||||
);
|
|
||||||
|
|
||||||
// financial statements
|
// financial statements
|
||||||
for (let report of [
|
for (let report of ['Trial Balance', 'General Ledger', 'Balance Sheet',
|
||||||
"Trial Balance",
|
'Profit and Loss Statement', 'Cash Flow Statement', 'Accounts Payable', 'Accounts Receivable']) {
|
||||||
"General Ledger",
|
treeview.page.add_inner_button(__(report), function() {
|
||||||
"Balance Sheet",
|
frappe.set_route('query-report', report, {company: get_company()});
|
||||||
"Profit and Loss Statement",
|
}, __('Financial Statements'));
|
||||||
"Cash Flow Statement",
|
|
||||||
"Accounts Payable",
|
|
||||||
"Accounts Receivable",
|
|
||||||
]) {
|
|
||||||
treeview.page.add_inner_button(
|
|
||||||
__(report),
|
|
||||||
function () {
|
|
||||||
frappe.set_route("query-report", report, { company: get_company() });
|
|
||||||
},
|
|
||||||
__("Financial Statements")
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
},
|
|
||||||
post_render: function (treeview) {
|
|
||||||
frappe.treeview_settings["Account"].treeview["tree"] = treeview.tree;
|
|
||||||
treeview.page.set_primary_action(
|
|
||||||
__("New"),
|
|
||||||
function () {
|
|
||||||
let root_company = treeview.page.fields_dict.root_company.get_value();
|
|
||||||
|
|
||||||
if (root_company) {
|
},
|
||||||
frappe.throw(__("Please add the account to root level Company - {0}"), [root_company]);
|
post_render: function(treeview) {
|
||||||
} else {
|
frappe.treeview_settings['Account'].treeview["tree"] = treeview.tree;
|
||||||
treeview.new_node();
|
treeview.page.set_primary_action(__("New"), function() {
|
||||||
}
|
let root_company = treeview.page.fields_dict.root_company.get_value();
|
||||||
},
|
|
||||||
"add"
|
if(root_company) {
|
||||||
);
|
frappe.throw(__("Please add the account to root level Company - {0}"), [root_company]);
|
||||||
|
} else {
|
||||||
|
treeview.new_node();
|
||||||
|
}
|
||||||
|
}, "add");
|
||||||
},
|
},
|
||||||
toolbar: [
|
toolbar: [
|
||||||
{
|
{
|
||||||
label: __("Add Child"),
|
label:__("Add Child"),
|
||||||
condition: function (node) {
|
condition: function(node) {
|
||||||
return (
|
return frappe.boot.user.can_create.indexOf("Account") !== -1
|
||||||
frappe.boot.user.can_create.indexOf("Account") !== -1 &&
|
&& (!frappe.treeview_settings['Account'].treeview.page.fields_dict.root_company.get_value()
|
||||||
(!frappe.treeview_settings[
|
|| frappe.flags.ignore_root_company_validation)
|
||||||
"Account"
|
&& node.expandable && !node.hide_add;
|
||||||
].treeview.page.fields_dict.root_company.get_value() ||
|
|
||||||
frappe.flags.ignore_root_company_validation) &&
|
|
||||||
node.expandable &&
|
|
||||||
!node.hide_add
|
|
||||||
);
|
|
||||||
},
|
},
|
||||||
click: function () {
|
click: function() {
|
||||||
var me = frappe.views.trees["Account"];
|
var me = frappe.views.trees['Account'];
|
||||||
me.new_node();
|
me.new_node();
|
||||||
},
|
},
|
||||||
btnClass: "hidden-xs",
|
btnClass: "hidden-xs"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
condition: function (node) {
|
condition: function(node) {
|
||||||
return !node.root && frappe.boot.user.can_read.indexOf("GL Entry") !== -1;
|
return !node.root && frappe.boot.user.can_read.indexOf("GL Entry") !== -1
|
||||||
},
|
},
|
||||||
label: __("View Ledger"),
|
label: __("View Ledger"),
|
||||||
click: function (node, btn) {
|
click: function(node, btn) {
|
||||||
frappe.route_options = {
|
frappe.route_options = {
|
||||||
account: node.label,
|
"account": node.label,
|
||||||
from_date: erpnext.utils.get_fiscal_year(frappe.datetime.get_today(), true)[1],
|
"from_date": erpnext.utils.get_fiscal_year(frappe.datetime.get_today(), true)[1],
|
||||||
to_date: erpnext.utils.get_fiscal_year(frappe.datetime.get_today(), true)[2],
|
"to_date": erpnext.utils.get_fiscal_year(frappe.datetime.get_today(), true)[2],
|
||||||
company:
|
"company": frappe.treeview_settings['Account'].treeview.page.fields_dict.company.get_value()
|
||||||
frappe.treeview_settings["Account"].treeview.page.fields_dict.company.get_value(),
|
|
||||||
};
|
};
|
||||||
frappe.set_route("query-report", "General Ledger");
|
frappe.set_route("query-report", "General Ledger");
|
||||||
},
|
},
|
||||||
btnClass: "hidden-xs",
|
btnClass: "hidden-xs"
|
||||||
},
|
}
|
||||||
],
|
],
|
||||||
extend_toolbar: true,
|
extend_toolbar: true
|
||||||
};
|
}
|
||||||
|
|||||||
@@ -74,7 +74,7 @@ def create_charts(
|
|||||||
# after all accounts are already inserted.
|
# after all accounts are already inserted.
|
||||||
frappe.local.flags.ignore_update_nsm = True
|
frappe.local.flags.ignore_update_nsm = True
|
||||||
_import_accounts(chart, None, None, root_account=True)
|
_import_accounts(chart, None, None, root_account=True)
|
||||||
rebuild_tree("Account", "parent_account")
|
rebuild_tree("Account")
|
||||||
frappe.local.flags.ignore_update_nsm = False
|
frappe.local.flags.ignore_update_nsm = False
|
||||||
|
|
||||||
|
|
||||||
@@ -231,6 +231,8 @@ def build_account_tree(tree, parent, all_accounts):
|
|||||||
tree[child.account_name]["account_type"] = child.account_type
|
tree[child.account_name]["account_type"] = child.account_type
|
||||||
if child.tax_rate:
|
if child.tax_rate:
|
||||||
tree[child.account_name]["tax_rate"] = child.tax_rate
|
tree[child.account_name]["tax_rate"] = child.tax_rate
|
||||||
|
if child.account_currency:
|
||||||
|
tree[child.account_name]["account_currency"] = child.account_currency
|
||||||
if not parent:
|
if not parent:
|
||||||
tree[child.account_name]["root_type"] = child.root_type
|
tree[child.account_name]["root_type"] = child.root_type
|
||||||
|
|
||||||
|
|||||||
@@ -56,9 +56,7 @@
|
|||||||
"Constru\u00e7\u00f5es em Andamento de Im\u00f3veis Destinados \u00e0 Venda": {},
|
"Constru\u00e7\u00f5es em Andamento de Im\u00f3veis Destinados \u00e0 Venda": {},
|
||||||
"Estoques Destinados \u00e0 Doa\u00e7\u00e3o": {},
|
"Estoques Destinados \u00e0 Doa\u00e7\u00e3o": {},
|
||||||
"Im\u00f3veis Destinados \u00e0 Venda": {},
|
"Im\u00f3veis Destinados \u00e0 Venda": {},
|
||||||
"Insumos (materiais diretos)": {
|
"Insumos (materiais diretos)": {},
|
||||||
"account_type": "Stock"
|
|
||||||
},
|
|
||||||
"Insumos Agropecu\u00e1rios": {},
|
"Insumos Agropecu\u00e1rios": {},
|
||||||
"Mercadorias para Revenda": {},
|
"Mercadorias para Revenda": {},
|
||||||
"Outras 11": {},
|
"Outras 11": {},
|
||||||
@@ -148,65 +146,6 @@
|
|||||||
"root_type": "Asset"
|
"root_type": "Asset"
|
||||||
},
|
},
|
||||||
"CUSTOS DE PRODU\u00c7\u00c3O": {
|
"CUSTOS DE PRODU\u00c7\u00c3O": {
|
||||||
"CUSTO DOS PRODUTOS E SERVI\u00c7OS VENDIDOS": {
|
|
||||||
"CUSTO DOS PRODUTOS VENDIDOS": {
|
|
||||||
"CUSTO DOS PRODUTOS VENDIDOS PARA AS DEMAIS ATIVIDADES": {
|
|
||||||
"Custos dos Produtos Vendidos em Geral": {
|
|
||||||
"account_type": "Cost of Goods Sold"
|
|
||||||
},
|
|
||||||
"Outros Custos 4": {},
|
|
||||||
"account_type": "Cost of Goods Sold"
|
|
||||||
},
|
|
||||||
"CUSTO DOS PRODUTOS VENDIDOS PARA ASSIST\u00caNCIA SOCIAL": {
|
|
||||||
"Custos dos Produtos para Assist\u00eancia Social - Gratuidades": {},
|
|
||||||
"Custos dos Produtos para Assist\u00eancia Social - Vendidos": {},
|
|
||||||
"Outras": {}
|
|
||||||
},
|
|
||||||
"CUSTO DOS PRODUTOS VENDIDOS PARA EDUCA\u00c7\u00c3O": {
|
|
||||||
"Custos dos Produtos para Educa\u00e7\u00e3o - Gratuidades": {},
|
|
||||||
"Custos dos Produtos para Educa\u00e7\u00e3o - Vendidos": {},
|
|
||||||
"Outros Custos 6": {}
|
|
||||||
},
|
|
||||||
"CUSTO DOS PRODUTOS VENDIDOS PARA SA\u00daDE": {
|
|
||||||
"Custos dos Produtos para Sa\u00fade - Gratuidades": {},
|
|
||||||
"Custos dos Produtos para Sa\u00fade \u2013 Vendidos": {},
|
|
||||||
"Outros Custos 5": {}
|
|
||||||
},
|
|
||||||
"account_type": "Cost of Goods Sold"
|
|
||||||
},
|
|
||||||
"CUSTO DOS SERVI\u00c7OS PRESTADOS": {
|
|
||||||
"CUSTO DOS SERVI\u00c7OS PRESTADOS PARA AS DEMAIS ATIVIDADES": {
|
|
||||||
"Custo dos Servi\u00e7os Prestados em Geral": {},
|
|
||||||
"Outros Custos": {}
|
|
||||||
},
|
|
||||||
"CUSTO DOS SERVI\u00c7OS PRESTADOS PARA ASSIST\u00caNCIA SOCIAL": {
|
|
||||||
"Custo dos Servi\u00e7os Prestados a Conv\u00eanios/Contratos/Parcerias": {},
|
|
||||||
"Custo dos Servi\u00e7os Prestados a Doa\u00e7\u00f5es 1": {},
|
|
||||||
"Custo dos Servi\u00e7os Prestados a Doa\u00e7\u00f5es/Subven\u00e7\u00f5es Vinculadas 1": {},
|
|
||||||
"Custo dos Servi\u00e7os Prestados a Gratuidade 1": {},
|
|
||||||
"Custo dos Servi\u00e7os Prestados a Pacientes Particulares": {},
|
|
||||||
"Outros Custos 2": {}
|
|
||||||
},
|
|
||||||
"CUSTO DOS SERVI\u00c7OS PRESTADOS PARA EDUCA\u00c7\u00c3O": {
|
|
||||||
"Custo dos Servi\u00e7os Prestados a Alunos N\u00e3o Bolsistas": {},
|
|
||||||
"Custo dos Servi\u00e7os Prestados a Conv\u00eanios/Contratos/Parcerias (Exceto PROUNI)": {},
|
|
||||||
"Custo dos Servi\u00e7os Prestados a Doa\u00e7\u00f5es": {},
|
|
||||||
"Custo dos Servi\u00e7os Prestados a Doa\u00e7\u00f5es/Subven\u00e7\u00f5es Vinculadas": {},
|
|
||||||
"Custo dos Servi\u00e7os Prestados a Gratuidade": {},
|
|
||||||
"Custo dos Servi\u00e7os Prestados ao PROUNI": {},
|
|
||||||
"Outros Custos 1": {}
|
|
||||||
},
|
|
||||||
"CUSTO DOS SERVI\u00c7OS PRESTADOS PARA SA\u00daDE": {
|
|
||||||
"Custo dos Servi\u00e7os Prestados a Conv\u00eanios SUS": {},
|
|
||||||
"Custo dos Servi\u00e7os Prestados a Conv\u00eanios/Contratos/Parcerias 1": {},
|
|
||||||
"Custo dos Servi\u00e7os Prestados a Doa\u00e7\u00f5es 2": {},
|
|
||||||
"Custo dos Servi\u00e7os Prestados a Doa\u00e7\u00f5es/Subven\u00e7\u00f5es Vinculadas 2": {},
|
|
||||||
"Custo dos Servi\u00e7os Prestados a Gratuidade 2": {},
|
|
||||||
"Custo dos Servi\u00e7os Prestados a Pacientes Particulares 1": {},
|
|
||||||
"Outros Custos 3": {}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"CUSTO DOS BENS E SERVI\u00c7OS PRODUZIDOS": {
|
"CUSTO DOS BENS E SERVI\u00c7OS PRODUZIDOS": {
|
||||||
"CUSTO DOS PRODUTOS DE FABRICA\u00c7\u00c3O PR\u00d3PRIA PRODUZIDOS": {
|
"CUSTO DOS PRODUTOS DE FABRICA\u00c7\u00c3O PR\u00d3PRIA PRODUZIDOS": {
|
||||||
"Alimenta\u00e7\u00e3o do Trabalhador": {},
|
"Alimenta\u00e7\u00e3o do Trabalhador": {},
|
||||||
@@ -682,9 +621,7 @@
|
|||||||
"Receita das Unidades Imobili\u00e1rias Vendidas": {},
|
"Receita das Unidades Imobili\u00e1rias Vendidas": {},
|
||||||
"Receita de Exporta\u00e7\u00e3o Direta de Mercadorias e Produtos": {},
|
"Receita de Exporta\u00e7\u00e3o Direta de Mercadorias e Produtos": {},
|
||||||
"Receita de Exporta\u00e7\u00e3o de Servi\u00e7os": {},
|
"Receita de Exporta\u00e7\u00e3o de Servi\u00e7os": {},
|
||||||
"Receita de Loca\u00e7\u00e3o de Bens M\u00f3veis e Im\u00f3veis": {
|
"Receita de Loca\u00e7\u00e3o de Bens M\u00f3veis e Im\u00f3veis": {},
|
||||||
"account_type": "Income Account"
|
|
||||||
},
|
|
||||||
"Receita de Vendas de Mercadorias e Produtos a Comercial Exportadora com Fim Espec\u00edfico de Exporta\u00e7\u00e3o": {}
|
"Receita de Vendas de Mercadorias e Produtos a Comercial Exportadora com Fim Espec\u00edfico de Exporta\u00e7\u00e3o": {}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -708,6 +645,65 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
"RESULTADO OPERACIONAL": {
|
"RESULTADO OPERACIONAL": {
|
||||||
|
"CUSTO DOS PRODUTOS E SERVI\u00c7OS VENDIDOS": {
|
||||||
|
"CUSTO DOS PRODUTOS VENDIDOS": {
|
||||||
|
"CUSTO DOS PRODUTOS VENDIDOS PARA AS DEMAIS ATIVIDADES": {
|
||||||
|
"Custos dos Produtos Vendidos em Geral": {
|
||||||
|
"account_type": "Cost of Goods Sold"
|
||||||
|
},
|
||||||
|
"Outros Custos 4": {},
|
||||||
|
"account_type": "Cost of Goods Sold"
|
||||||
|
},
|
||||||
|
"CUSTO DOS PRODUTOS VENDIDOS PARA ASSIST\u00caNCIA SOCIAL": {
|
||||||
|
"Custos dos Produtos para Assist\u00eancia Social - Gratuidades": {},
|
||||||
|
"Custos dos Produtos para Assist\u00eancia Social - Vendidos": {},
|
||||||
|
"Outras": {}
|
||||||
|
},
|
||||||
|
"CUSTO DOS PRODUTOS VENDIDOS PARA EDUCA\u00c7\u00c3O": {
|
||||||
|
"Custos dos Produtos para Educa\u00e7\u00e3o - Gratuidades": {},
|
||||||
|
"Custos dos Produtos para Educa\u00e7\u00e3o - Vendidos": {},
|
||||||
|
"Outros Custos 6": {}
|
||||||
|
},
|
||||||
|
"CUSTO DOS PRODUTOS VENDIDOS PARA SA\u00daDE": {
|
||||||
|
"Custos dos Produtos para Sa\u00fade - Gratuidades": {},
|
||||||
|
"Custos dos Produtos para Sa\u00fade \u2013 Vendidos": {},
|
||||||
|
"Outros Custos 5": {}
|
||||||
|
},
|
||||||
|
"account_type": "Cost of Goods Sold"
|
||||||
|
},
|
||||||
|
"CUSTO DOS SERVI\u00c7OS PRESTADOS": {
|
||||||
|
"CUSTO DOS SERVI\u00c7OS PRESTADOS PARA AS DEMAIS ATIVIDADES": {
|
||||||
|
"Custo dos Servi\u00e7os Prestados em Geral": {},
|
||||||
|
"Outros Custos": {}
|
||||||
|
},
|
||||||
|
"CUSTO DOS SERVI\u00c7OS PRESTADOS PARA ASSIST\u00caNCIA SOCIAL": {
|
||||||
|
"Custo dos Servi\u00e7os Prestados a Conv\u00eanios/Contratos/Parcerias": {},
|
||||||
|
"Custo dos Servi\u00e7os Prestados a Doa\u00e7\u00f5es 1": {},
|
||||||
|
"Custo dos Servi\u00e7os Prestados a Doa\u00e7\u00f5es/Subven\u00e7\u00f5es Vinculadas 1": {},
|
||||||
|
"Custo dos Servi\u00e7os Prestados a Gratuidade 1": {},
|
||||||
|
"Custo dos Servi\u00e7os Prestados a Pacientes Particulares": {},
|
||||||
|
"Outros Custos 2": {}
|
||||||
|
},
|
||||||
|
"CUSTO DOS SERVI\u00c7OS PRESTADOS PARA EDUCA\u00c7\u00c3O": {
|
||||||
|
"Custo dos Servi\u00e7os Prestados a Alunos N\u00e3o Bolsistas": {},
|
||||||
|
"Custo dos Servi\u00e7os Prestados a Conv\u00eanios/Contratos/Parcerias (Exceto PROUNI)": {},
|
||||||
|
"Custo dos Servi\u00e7os Prestados a Doa\u00e7\u00f5es": {},
|
||||||
|
"Custo dos Servi\u00e7os Prestados a Doa\u00e7\u00f5es/Subven\u00e7\u00f5es Vinculadas": {},
|
||||||
|
"Custo dos Servi\u00e7os Prestados a Gratuidade": {},
|
||||||
|
"Custo dos Servi\u00e7os Prestados ao PROUNI": {},
|
||||||
|
"Outros Custos 1": {}
|
||||||
|
},
|
||||||
|
"CUSTO DOS SERVI\u00c7OS PRESTADOS PARA SA\u00daDE": {
|
||||||
|
"Custo dos Servi\u00e7os Prestados a Conv\u00eanios SUS": {},
|
||||||
|
"Custo dos Servi\u00e7os Prestados a Conv\u00eanios/Contratos/Parcerias 1": {},
|
||||||
|
"Custo dos Servi\u00e7os Prestados a Doa\u00e7\u00f5es 2": {},
|
||||||
|
"Custo dos Servi\u00e7os Prestados a Doa\u00e7\u00f5es/Subven\u00e7\u00f5es Vinculadas 2": {},
|
||||||
|
"Custo dos Servi\u00e7os Prestados a Gratuidade 2": {},
|
||||||
|
"Custo dos Servi\u00e7os Prestados a Pacientes Particulares 1": {},
|
||||||
|
"Outros Custos 3": {}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
"DESPESAS OPERACIONAIS": {
|
"DESPESAS OPERACIONAIS": {
|
||||||
"DESPESAS OPERACIONAIS 1": {
|
"DESPESAS OPERACIONAIS 1": {
|
||||||
"DESPESAS OPERACIONAIS 2": {
|
"DESPESAS OPERACIONAIS 2": {
|
||||||
|
|||||||
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
@@ -1653,4 +1653,4 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"country_code": "ni",
|
"country_code": "ni",
|
||||||
"name": "Nicaragua - Catalogo de Cuentas",
|
"name": "Nicaragua - Catálogo de Cuentas",
|
||||||
"tree": {
|
"tree": {
|
||||||
"Activo": {
|
"Activo": {
|
||||||
"Activo Corriente": {
|
"Activo Corriente": {
|
||||||
@@ -491,4 +491,4 @@
|
|||||||
"root_type": "Liability"
|
"root_type": "Liability"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -6,7 +6,6 @@ import unittest
|
|||||||
|
|
||||||
import frappe
|
import frappe
|
||||||
from frappe.test_runner import make_test_records
|
from frappe.test_runner import make_test_records
|
||||||
from frappe.utils import nowdate
|
|
||||||
|
|
||||||
from erpnext.accounts.doctype.account.account import (
|
from erpnext.accounts.doctype.account.account import (
|
||||||
InvalidAccountMergeError,
|
InvalidAccountMergeError,
|
||||||
@@ -325,19 +324,6 @@ class TestAccount(unittest.TestCase):
|
|||||||
acc.account_currency = "USD"
|
acc.account_currency = "USD"
|
||||||
self.assertRaises(frappe.ValidationError, acc.save)
|
self.assertRaises(frappe.ValidationError, acc.save)
|
||||||
|
|
||||||
def test_account_balance(self):
|
|
||||||
from erpnext.accounts.utils import get_balance_on
|
|
||||||
|
|
||||||
if not frappe.db.exists("Account", "Test Percent Account %5 - _TC"):
|
|
||||||
acc = frappe.new_doc("Account")
|
|
||||||
acc.account_name = "Test Percent Account %5"
|
|
||||||
acc.parent_account = "Tax Assets - _TC"
|
|
||||||
acc.company = "_Test Company"
|
|
||||||
acc.insert()
|
|
||||||
|
|
||||||
balance = get_balance_on(account="Test Percent Account %5 - _TC", date=nowdate())
|
|
||||||
self.assertEqual(balance, 0)
|
|
||||||
|
|
||||||
|
|
||||||
def _make_test_records(verbose=None):
|
def _make_test_records(verbose=None):
|
||||||
from frappe.test_runner import make_test_objects
|
from frappe.test_runner import make_test_objects
|
||||||
|
|||||||
@@ -1,86 +1,74 @@
|
|||||||
// Copyright (c) 2019, Frappe Technologies Pvt. Ltd. and contributors
|
// Copyright (c) 2019, Frappe Technologies Pvt. Ltd. and contributors
|
||||||
// For license information, please see license.txt
|
// For license information, please see license.txt
|
||||||
|
|
||||||
frappe.ui.form.on("Accounting Dimension", {
|
frappe.ui.form.on('Accounting Dimension', {
|
||||||
refresh: function (frm) {
|
refresh: function(frm) {
|
||||||
frm.set_query("document_type", () => {
|
frm.set_query('document_type', () => {
|
||||||
let invalid_doctypes = frappe.model.core_doctypes_list;
|
let invalid_doctypes = frappe.model.core_doctypes_list;
|
||||||
invalid_doctypes.push(
|
invalid_doctypes.push('Accounting Dimension', 'Project',
|
||||||
"Accounting Dimension",
|
'Cost Center', 'Accounting Dimension Detail', 'Company');
|
||||||
"Project",
|
|
||||||
"Cost Center",
|
|
||||||
"Accounting Dimension Detail",
|
|
||||||
"Company"
|
|
||||||
);
|
|
||||||
|
|
||||||
return {
|
return {
|
||||||
filters: {
|
filters: {
|
||||||
name: ["not in", invalid_doctypes],
|
name: ['not in', invalid_doctypes]
|
||||||
},
|
}
|
||||||
};
|
};
|
||||||
});
|
});
|
||||||
|
|
||||||
frm.set_query("offsetting_account", "dimension_defaults", function (doc, cdt, cdn) {
|
frm.set_query("offsetting_account", "dimension_defaults", function(doc, cdt, cdn) {
|
||||||
let d = locals[cdt][cdn];
|
let d = locals[cdt][cdn];
|
||||||
return {
|
return {
|
||||||
filters: {
|
filters: {
|
||||||
company: d.company,
|
company: d.company,
|
||||||
root_type: ["in", ["Asset", "Liability"]],
|
root_type: ["in", ["Asset", "Liability"]],
|
||||||
is_group: 0,
|
is_group: 0
|
||||||
},
|
}
|
||||||
};
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
if (!frm.is_new()) {
|
if (!frm.is_new()) {
|
||||||
frm.add_custom_button(__("Show {0}", [frm.doc.document_type]), function () {
|
frm.add_custom_button(__('Show {0}', [frm.doc.document_type]), function () {
|
||||||
frappe.set_route("List", frm.doc.document_type);
|
frappe.set_route("List", frm.doc.document_type);
|
||||||
});
|
});
|
||||||
|
|
||||||
let button = frm.doc.disabled ? "Enable" : "Disable";
|
let button = frm.doc.disabled ? "Enable" : "Disable";
|
||||||
|
|
||||||
frm.add_custom_button(__(button), function () {
|
frm.add_custom_button(__(button), function() {
|
||||||
frm.set_value("disabled", 1 - frm.doc.disabled);
|
|
||||||
|
frm.set_value('disabled', 1 - frm.doc.disabled);
|
||||||
|
|
||||||
frappe.call({
|
frappe.call({
|
||||||
method: "erpnext.accounts.doctype.accounting_dimension.accounting_dimension.disable_dimension",
|
method: "erpnext.accounts.doctype.accounting_dimension.accounting_dimension.disable_dimension",
|
||||||
args: {
|
args: {
|
||||||
doc: frm.doc,
|
doc: frm.doc
|
||||||
},
|
},
|
||||||
freeze: true,
|
freeze: true,
|
||||||
callback: function (r) {
|
callback: function(r) {
|
||||||
let message = frm.doc.disabled ? "Dimension Disabled" : "Dimension Enabled";
|
let message = frm.doc.disabled ? "Dimension Disabled" : "Dimension Enabled";
|
||||||
frm.save();
|
frm.save();
|
||||||
frappe.show_alert({ message: __(message), indicator: "green" });
|
frappe.show_alert({message:__(message), indicator:'green'});
|
||||||
},
|
}
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
document_type: function (frm) {
|
document_type: function(frm) {
|
||||||
frm.set_value("label", frm.doc.document_type);
|
|
||||||
frm.set_value("fieldname", frappe.model.scrub(frm.doc.document_type));
|
|
||||||
|
|
||||||
frappe.db.get_value(
|
frm.set_value('label', frm.doc.document_type);
|
||||||
"Accounting Dimension",
|
frm.set_value('fieldname', frappe.model.scrub(frm.doc.document_type));
|
||||||
{ document_type: frm.doc.document_type },
|
|
||||||
"document_type",
|
frappe.db.get_value('Accounting Dimension', {'document_type': frm.doc.document_type}, 'document_type', (r) => {
|
||||||
(r) => {
|
if (r && r.document_type) {
|
||||||
if (r && r.document_type) {
|
frm.set_df_property('document_type', 'description', "Document type is already set as dimension");
|
||||||
frm.set_df_property(
|
|
||||||
"document_type",
|
|
||||||
"description",
|
|
||||||
"Document type is already set as dimension"
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
);
|
});
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
frappe.ui.form.on("Accounting Dimension Detail", {
|
frappe.ui.form.on('Accounting Dimension Detail', {
|
||||||
dimension_defaults_add: function (frm, cdt, cdn) {
|
dimension_defaults_add: function(frm, cdt, cdn) {
|
||||||
let row = locals[cdt][cdn];
|
let row = locals[cdt][cdn];
|
||||||
row.reference_document = frm.doc.document_type;
|
row.reference_document = frm.doc.document_type;
|
||||||
},
|
}
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -11,10 +11,6 @@ from frappe.model import core_doctypes_list
|
|||||||
from frappe.model.document import Document
|
from frappe.model.document import Document
|
||||||
from frappe.utils import cstr
|
from frappe.utils import cstr
|
||||||
|
|
||||||
from erpnext.accounts.doctype.repost_accounting_ledger.repost_accounting_ledger import (
|
|
||||||
get_allowed_types_from_settings,
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
class AccountingDimension(Document):
|
class AccountingDimension(Document):
|
||||||
# begin: auto-generated types
|
# begin: auto-generated types
|
||||||
@@ -110,7 +106,6 @@ def make_dimension_in_accounting_doctypes(doc, doclist=None):
|
|||||||
|
|
||||||
doc_count = len(get_accounting_dimensions())
|
doc_count = len(get_accounting_dimensions())
|
||||||
count = 0
|
count = 0
|
||||||
repostable_doctypes = get_allowed_types_from_settings()
|
|
||||||
|
|
||||||
for doctype in doclist:
|
for doctype in doclist:
|
||||||
|
|
||||||
@@ -126,7 +121,6 @@ def make_dimension_in_accounting_doctypes(doc, doclist=None):
|
|||||||
"options": doc.document_type,
|
"options": doc.document_type,
|
||||||
"insert_after": insert_after_field,
|
"insert_after": insert_after_field,
|
||||||
"owner": "Administrator",
|
"owner": "Administrator",
|
||||||
"allow_on_submit": 1 if doctype in repostable_doctypes else 0,
|
|
||||||
}
|
}
|
||||||
|
|
||||||
meta = frappe.get_meta(doctype, cached=False)
|
meta = frappe.get_meta(doctype, cached=False)
|
||||||
|
|||||||
@@ -1,9 +1,10 @@
|
|||||||
// Copyright (c) 2020, Frappe Technologies Pvt. Ltd. and contributors
|
// Copyright (c) 2020, Frappe Technologies Pvt. Ltd. and contributors
|
||||||
// For license information, please see license.txt
|
// For license information, please see license.txt
|
||||||
|
|
||||||
frappe.ui.form.on("Accounting Dimension Filter", {
|
frappe.ui.form.on('Accounting Dimension Filter', {
|
||||||
refresh: function (frm, cdt, cdn) {
|
refresh: function(frm, cdt, cdn) {
|
||||||
let help_content = `<table class="table table-bordered" style="background-color: var(--scrollbar-track-color);">
|
let help_content =
|
||||||
|
`<table class="table table-bordered" style="background-color: var(--scrollbar-track-color);">
|
||||||
<tr><td>
|
<tr><td>
|
||||||
<p>
|
<p>
|
||||||
<i class="fa fa-hand-right"></i>
|
<i class="fa fa-hand-right"></i>
|
||||||
@@ -12,80 +13,77 @@ frappe.ui.form.on("Accounting Dimension Filter", {
|
|||||||
</td></tr>
|
</td></tr>
|
||||||
</table>`;
|
</table>`;
|
||||||
|
|
||||||
frm.set_df_property("dimension_filter_help", "options", help_content);
|
frm.set_df_property('dimension_filter_help', 'options', help_content);
|
||||||
},
|
},
|
||||||
onload: function (frm) {
|
onload: function(frm) {
|
||||||
frm.set_query("applicable_on_account", "accounts", function () {
|
frm.set_query('applicable_on_account', 'accounts', function() {
|
||||||
return {
|
return {
|
||||||
filters: {
|
filters: {
|
||||||
company: frm.doc.company,
|
'company': frm.doc.company
|
||||||
},
|
}
|
||||||
};
|
};
|
||||||
});
|
});
|
||||||
|
|
||||||
frappe.db.get_list("Accounting Dimension", { fields: ["document_type"] }).then((res) => {
|
frappe.db.get_list('Accounting Dimension',
|
||||||
let options = ["Cost Center", "Project"];
|
{fields: ['document_type']}).then((res) => {
|
||||||
|
let options = ['Cost Center', 'Project'];
|
||||||
|
|
||||||
res.forEach((dimension) => {
|
res.forEach((dimension) => {
|
||||||
options.push(dimension.document_type);
|
options.push(dimension.document_type);
|
||||||
});
|
});
|
||||||
|
|
||||||
frm.set_df_property("accounting_dimension", "options", options);
|
frm.set_df_property('accounting_dimension', 'options', options);
|
||||||
});
|
});
|
||||||
|
|
||||||
frm.trigger("setup_filters");
|
frm.trigger('setup_filters');
|
||||||
},
|
},
|
||||||
|
|
||||||
setup_filters: function (frm) {
|
setup_filters: function(frm) {
|
||||||
let filters = {};
|
let filters = {};
|
||||||
|
|
||||||
if (frm.doc.accounting_dimension) {
|
if (frm.doc.accounting_dimension) {
|
||||||
frappe.model.with_doctype(frm.doc.accounting_dimension, function () {
|
frappe.model.with_doctype(frm.doc.accounting_dimension, function() {
|
||||||
if (frappe.model.is_tree(frm.doc.accounting_dimension)) {
|
if (frappe.model.is_tree(frm.doc.accounting_dimension)) {
|
||||||
filters["is_group"] = 0;
|
filters['is_group'] = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (frappe.meta.has_field(frm.doc.accounting_dimension, "company")) {
|
if (frappe.meta.has_field(frm.doc.accounting_dimension, 'company')) {
|
||||||
filters["company"] = frm.doc.company;
|
filters['company'] = frm.doc.company;
|
||||||
}
|
}
|
||||||
|
|
||||||
frm.set_query("dimension_value", "dimensions", function () {
|
frm.set_query('dimension_value', 'dimensions', function() {
|
||||||
return {
|
return {
|
||||||
filters: filters,
|
filters: filters
|
||||||
};
|
};
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
accounting_dimension: function (frm) {
|
accounting_dimension: function(frm) {
|
||||||
frm.clear_table("dimensions");
|
frm.clear_table("dimensions");
|
||||||
let row = frm.add_child("dimensions");
|
let row = frm.add_child("dimensions");
|
||||||
row.accounting_dimension = frm.doc.accounting_dimension;
|
row.accounting_dimension = frm.doc.accounting_dimension;
|
||||||
frm.fields_dict["dimensions"].grid.update_docfield_property(
|
frm.fields_dict["dimensions"].grid.update_docfield_property("dimension_value", "label", frm.doc.accounting_dimension);
|
||||||
"dimension_value",
|
|
||||||
"label",
|
|
||||||
frm.doc.accounting_dimension
|
|
||||||
);
|
|
||||||
frm.refresh_field("dimensions");
|
frm.refresh_field("dimensions");
|
||||||
frm.trigger("setup_filters");
|
frm.trigger('setup_filters');
|
||||||
},
|
},
|
||||||
apply_restriction_on_values: function (frm) {
|
apply_restriction_on_values: function(frm) {
|
||||||
/** If restriction on values is not applied, we should set "allow_or_restrict" to "Restrict" with an empty allowed dimension table.
|
/** If restriction on values is not applied, we should set "allow_or_restrict" to "Restrict" with an empty allowed dimension table.
|
||||||
* Hence it's not "restricted" on any value.
|
* Hence it's not "restricted" on any value.
|
||||||
*/
|
*/
|
||||||
if (!frm.doc.apply_restriction_on_values) {
|
if (!frm.doc.apply_restriction_on_values) {
|
||||||
frm.set_value("allow_or_restrict", "Restrict");
|
frm.set_value("allow_or_restrict", "Restrict");
|
||||||
frm.clear_table("dimensions");
|
frm.clear_table("dimensions");
|
||||||
frm.refresh_field("dimensions");
|
frm.refresh_field("dimensions");
|
||||||
}
|
}
|
||||||
},
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
frappe.ui.form.on("Allowed Dimension", {
|
frappe.ui.form.on('Allowed Dimension', {
|
||||||
dimensions_add: function (frm, cdt, cdn) {
|
dimensions_add: function(frm, cdt, cdn) {
|
||||||
let row = locals[cdt][cdn];
|
let row = locals[cdt][cdn];
|
||||||
row.accounting_dimension = frm.doc.accounting_dimension;
|
row.accounting_dimension = frm.doc.accounting_dimension;
|
||||||
frm.refresh_field("dimensions");
|
frm.refresh_field("dimensions");
|
||||||
},
|
}
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -1,33 +1,30 @@
|
|||||||
// Copyright (c) 2018, Frappe Technologies Pvt. Ltd. and contributors
|
// Copyright (c) 2018, Frappe Technologies Pvt. Ltd. and contributors
|
||||||
// For license information, please see license.txt
|
// For license information, please see license.txt
|
||||||
|
|
||||||
frappe.ui.form.on("Accounting Period", {
|
frappe.ui.form.on('Accounting Period', {
|
||||||
onload: function (frm) {
|
onload: function(frm) {
|
||||||
if (
|
if(frm.doc.closed_documents.length === 0 || (frm.doc.closed_documents.length === 1 && frm.doc.closed_documents[0].document_type == undefined)) {
|
||||||
frm.doc.closed_documents.length === 0 ||
|
|
||||||
(frm.doc.closed_documents.length === 1 && frm.doc.closed_documents[0].document_type == undefined)
|
|
||||||
) {
|
|
||||||
frappe.call({
|
frappe.call({
|
||||||
method: "get_doctypes_for_closing",
|
method: "get_doctypes_for_closing",
|
||||||
doc: frm.doc,
|
doc:frm.doc,
|
||||||
callback: function (r) {
|
callback: function(r) {
|
||||||
if (r.message) {
|
if(r.message) {
|
||||||
cur_frm.clear_table("closed_documents");
|
cur_frm.clear_table("closed_documents");
|
||||||
r.message.forEach(function (element) {
|
r.message.forEach(function(element) {
|
||||||
var c = frm.add_child("closed_documents");
|
var c = frm.add_child("closed_documents");
|
||||||
c.document_type = element.document_type;
|
c.document_type = element.document_type;
|
||||||
c.closed = element.closed;
|
c.closed = element.closed;
|
||||||
});
|
});
|
||||||
refresh_field("closed_documents");
|
refresh_field("closed_documents");
|
||||||
}
|
}
|
||||||
},
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
frm.set_query("document_type", "closed_documents", () => {
|
frm.set_query("document_type", "closed_documents", () => {
|
||||||
return {
|
return {
|
||||||
query: "erpnext.controllers.queries.get_doctypes_for_closing",
|
query: "erpnext.controllers.queries.get_doctypes_for_closing",
|
||||||
};
|
}
|
||||||
});
|
});
|
||||||
},
|
}
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -1,6 +1,8 @@
|
|||||||
// Copyright (c) 2016, Frappe Technologies Pvt. Ltd. and contributors
|
// Copyright (c) 2016, Frappe Technologies Pvt. Ltd. and contributors
|
||||||
// For license information, please see license.txt
|
// For license information, please see license.txt
|
||||||
|
|
||||||
frappe.ui.form.on("Accounts Settings", {
|
frappe.ui.form.on('Accounts Settings', {
|
||||||
refresh: function (frm) {},
|
refresh: function(frm) {
|
||||||
|
|
||||||
|
}
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
{
|
{
|
||||||
"actions": [],
|
"actions": [],
|
||||||
"creation": "2013-06-24 15:49:57",
|
"creation": "2013-06-24 15:49:57",
|
||||||
|
"description": "Settings for Accounts",
|
||||||
"doctype": "DocType",
|
"doctype": "DocType",
|
||||||
"document_type": "Other",
|
"document_type": "Other",
|
||||||
"editable_grid": 1,
|
"editable_grid": 1,
|
||||||
@@ -461,7 +462,7 @@
|
|||||||
"index_web_pages_for_search": 1,
|
"index_web_pages_for_search": 1,
|
||||||
"issingle": 1,
|
"issingle": 1,
|
||||||
"links": [],
|
"links": [],
|
||||||
"modified": "2024-01-30 14:04:26.553554",
|
"modified": "2023-11-20 09:37:47.650347",
|
||||||
"modified_by": "Administrator",
|
"modified_by": "Administrator",
|
||||||
"module": "Accounts",
|
"module": "Accounts",
|
||||||
"name": "Accounts Settings",
|
"name": "Accounts Settings",
|
||||||
|
|||||||
@@ -1,11 +1,8 @@
|
|||||||
frappe.ui.form.on("Accounts Settings", {
|
|
||||||
refresh: function (frm) {
|
frappe.ui.form.on('Accounts Settings', {
|
||||||
|
refresh: function(frm) {
|
||||||
frm.set_df_property("acc_frozen_upto", "label", "Books Closed Through");
|
frm.set_df_property("acc_frozen_upto", "label", "Books Closed Through");
|
||||||
frm.set_df_property(
|
frm.set_df_property("frozen_accounts_modifier", "label", "Role Allowed to Close Books & Make Changes to Closed Periods");
|
||||||
"frozen_accounts_modifier",
|
|
||||||
"label",
|
|
||||||
"Role Allowed to Close Books & Make Changes to Closed Periods"
|
|
||||||
);
|
|
||||||
frm.set_df_property("credit_controller", "label", "Credit Manager");
|
frm.set_df_property("credit_controller", "label", "Credit Manager");
|
||||||
},
|
}
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -1,36 +1,38 @@
|
|||||||
// Copyright (c) 2018, Frappe Technologies Pvt. Ltd. and contributors
|
// Copyright (c) 2018, Frappe Technologies Pvt. Ltd. and contributors
|
||||||
// For license information, please see license.txt
|
// For license information, please see license.txt
|
||||||
frappe.provide("erpnext.integrations");
|
frappe.provide('erpnext.integrations');
|
||||||
|
|
||||||
frappe.ui.form.on("Bank", {
|
frappe.ui.form.on('Bank', {
|
||||||
onload: function (frm) {
|
onload: function(frm) {
|
||||||
add_fields_to_mapping_table(frm);
|
add_fields_to_mapping_table(frm);
|
||||||
},
|
},
|
||||||
refresh: function (frm) {
|
refresh: function(frm) {
|
||||||
add_fields_to_mapping_table(frm);
|
add_fields_to_mapping_table(frm);
|
||||||
frm.toggle_display(["address_html", "contact_html"], !frm.doc.__islocal);
|
frm.toggle_display(['address_html','contact_html'], !frm.doc.__islocal);
|
||||||
|
|
||||||
if (frm.doc.__islocal) {
|
if (frm.doc.__islocal) {
|
||||||
frm.set_df_property("address_and_contact", "hidden", 1);
|
frm.set_df_property('address_and_contact', 'hidden', 1);
|
||||||
frappe.contacts.clear_address_and_contact(frm);
|
frappe.contacts.clear_address_and_contact(frm);
|
||||||
} else {
|
}
|
||||||
frm.set_df_property("address_and_contact", "hidden", 0);
|
else {
|
||||||
|
frm.set_df_property('address_and_contact', 'hidden', 0);
|
||||||
frappe.contacts.render_address_and_contact(frm);
|
frappe.contacts.render_address_and_contact(frm);
|
||||||
}
|
}
|
||||||
if (frm.doc.plaid_access_token) {
|
if (frm.doc.plaid_access_token) {
|
||||||
frm.add_custom_button(__("Refresh Plaid Link"), () => {
|
frm.add_custom_button(__('Refresh Plaid Link'), () => {
|
||||||
new erpnext.integrations.refreshPlaidLink(frm.doc.plaid_access_token);
|
new erpnext.integrations.refreshPlaidLink(frm.doc.plaid_access_token);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
},
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
||||||
let add_fields_to_mapping_table = function (frm) {
|
let add_fields_to_mapping_table = function (frm) {
|
||||||
let options = [];
|
let options = [];
|
||||||
|
|
||||||
frappe.model.with_doctype("Bank Transaction", function () {
|
frappe.model.with_doctype("Bank Transaction", function() {
|
||||||
let meta = frappe.get_meta("Bank Transaction");
|
let meta = frappe.get_meta("Bank Transaction");
|
||||||
meta.fields.forEach((value) => {
|
meta.fields.forEach(value => {
|
||||||
if (!["Section Break", "Column Break"].includes(value.fieldtype)) {
|
if (!["Section Break", "Column Break"].includes(value.fieldtype)) {
|
||||||
options.push(value.fieldname);
|
options.push(value.fieldname);
|
||||||
}
|
}
|
||||||
@@ -38,32 +40,30 @@ let add_fields_to_mapping_table = function (frm) {
|
|||||||
});
|
});
|
||||||
|
|
||||||
frm.fields_dict.bank_transaction_mapping.grid.update_docfield_property(
|
frm.fields_dict.bank_transaction_mapping.grid.update_docfield_property(
|
||||||
"bank_transaction_field",
|
'bank_transaction_field', 'options', options
|
||||||
"options",
|
|
||||||
options
|
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
erpnext.integrations.refreshPlaidLink = class refreshPlaidLink {
|
erpnext.integrations.refreshPlaidLink = class refreshPlaidLink {
|
||||||
constructor(access_token) {
|
constructor(access_token) {
|
||||||
this.access_token = access_token;
|
this.access_token = access_token;
|
||||||
this.plaidUrl = "https://cdn.plaid.com/link/v2/stable/link-initialize.js";
|
this.plaidUrl = 'https://cdn.plaid.com/link/v2/stable/link-initialize.js';
|
||||||
this.init_config();
|
this.init_config();
|
||||||
}
|
}
|
||||||
|
|
||||||
async init_config() {
|
async init_config() {
|
||||||
this.plaid_env = await frappe.db.get_single_value("Plaid Settings", "plaid_env");
|
this.plaid_env = await frappe.db.get_single_value('Plaid Settings', 'plaid_env');
|
||||||
this.token = await this.get_link_token_for_update();
|
this.token = await this.get_link_token_for_update();
|
||||||
this.init_plaid();
|
this.init_plaid();
|
||||||
}
|
}
|
||||||
|
|
||||||
async get_link_token_for_update() {
|
async get_link_token_for_update() {
|
||||||
const token = frappe.xcall(
|
const token = frappe.xcall(
|
||||||
"erpnext.erpnext_integrations.doctype.plaid_settings.plaid_settings.get_link_token_for_update",
|
'erpnext.erpnext_integrations.doctype.plaid_settings.plaid_settings.get_link_token_for_update',
|
||||||
{ access_token: this.access_token }
|
{ access_token: this.access_token }
|
||||||
);
|
)
|
||||||
if (!token) {
|
if (!token) {
|
||||||
frappe.throw(__("Cannot retrieve link token for update. Check Error Log for more information"));
|
frappe.throw(__('Cannot retrieve link token for update. Check Error Log for more information'));
|
||||||
}
|
}
|
||||||
return token;
|
return token;
|
||||||
}
|
}
|
||||||
@@ -90,45 +90,35 @@ erpnext.integrations.refreshPlaidLink = class refreshPlaidLink {
|
|||||||
resolve();
|
resolve();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
const el = document.createElement("script");
|
const el = document.createElement('script');
|
||||||
el.type = "text/javascript";
|
el.type = 'text/javascript';
|
||||||
el.async = true;
|
el.async = true;
|
||||||
el.src = src;
|
el.src = src;
|
||||||
el.addEventListener("load", resolve);
|
el.addEventListener('load', resolve);
|
||||||
el.addEventListener("error", reject);
|
el.addEventListener('error', reject);
|
||||||
el.addEventListener("abort", reject);
|
el.addEventListener('abort', reject);
|
||||||
document.head.appendChild(el);
|
document.head.appendChild(el);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
onScriptLoaded(me) {
|
onScriptLoaded(me) {
|
||||||
me.linkHandler = Plaid.create({
|
me.linkHandler = Plaid.create({ // eslint-disable-line no-undef
|
||||||
// eslint-disable-line no-undef
|
|
||||||
env: me.plaid_env,
|
env: me.plaid_env,
|
||||||
token: me.token,
|
token: me.token,
|
||||||
onSuccess: me.plaid_success,
|
onSuccess: me.plaid_success
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
onScriptError(error) {
|
onScriptError(error) {
|
||||||
frappe.msgprint(
|
frappe.msgprint(__("There was an issue connecting to Plaid's authentication server. Check browser console for more information"));
|
||||||
__(
|
|
||||||
"There was an issue connecting to Plaid's authentication server. Check browser console for more information"
|
|
||||||
)
|
|
||||||
);
|
|
||||||
console.log(error);
|
console.log(error);
|
||||||
}
|
}
|
||||||
|
|
||||||
plaid_success(token, response) {
|
plaid_success(token, response) {
|
||||||
frappe
|
frappe.xcall('erpnext.erpnext_integrations.doctype.plaid_settings.plaid_settings.update_bank_account_ids', {
|
||||||
.xcall(
|
response: response,
|
||||||
"erpnext.erpnext_integrations.doctype.plaid_settings.plaid_settings.update_bank_account_ids",
|
}).then(() => {
|
||||||
{
|
frappe.show_alert({ message: __('Plaid Link Updated'), indicator: 'green' });
|
||||||
response: response,
|
});
|
||||||
}
|
|
||||||
)
|
|
||||||
.then(() => {
|
|
||||||
frappe.show_alert({ message: __("Plaid Link Updated"), indicator: "green" });
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -1,49 +1,45 @@
|
|||||||
// Copyright (c) 2018, Frappe Technologies Pvt. Ltd. and contributors
|
// Copyright (c) 2018, Frappe Technologies Pvt. Ltd. and contributors
|
||||||
// For license information, please see license.txt
|
// For license information, please see license.txt
|
||||||
|
|
||||||
frappe.ui.form.on("Bank Account", {
|
frappe.ui.form.on('Bank Account', {
|
||||||
setup: function (frm) {
|
setup: function(frm) {
|
||||||
frm.set_query("account", function () {
|
frm.set_query("account", function() {
|
||||||
return {
|
return {
|
||||||
filters: {
|
filters: {
|
||||||
account_type: "Bank",
|
'account_type': 'Bank',
|
||||||
company: frm.doc.company,
|
'company': frm.doc.company,
|
||||||
is_group: 0,
|
'is_group': 0
|
||||||
},
|
}
|
||||||
};
|
};
|
||||||
});
|
});
|
||||||
frm.set_query("party_type", function () {
|
frm.set_query("party_type", function() {
|
||||||
return {
|
return {
|
||||||
query: "erpnext.setup.doctype.party_type.party_type.get_party_type",
|
query: "erpnext.setup.doctype.party_type.party_type.get_party_type",
|
||||||
};
|
};
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
refresh: function (frm) {
|
refresh: function(frm) {
|
||||||
frappe.dynamic_link = { doc: frm.doc, fieldname: "name", doctype: "Bank Account" };
|
frappe.dynamic_link = { doc: frm.doc, fieldname: 'name', doctype: 'Bank Account' }
|
||||||
|
|
||||||
frm.toggle_display(["address_html", "contact_html"], !frm.doc.__islocal);
|
frm.toggle_display(['address_html','contact_html'], !frm.doc.__islocal);
|
||||||
|
|
||||||
if (frm.doc.__islocal) {
|
if (frm.doc.__islocal) {
|
||||||
frappe.contacts.clear_address_and_contact(frm);
|
frappe.contacts.clear_address_and_contact(frm);
|
||||||
} else {
|
}
|
||||||
|
else {
|
||||||
frappe.contacts.render_address_and_contact(frm);
|
frappe.contacts.render_address_and_contact(frm);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (frm.doc.integration_id) {
|
if (frm.doc.integration_id) {
|
||||||
frm.add_custom_button(__("Unlink external integrations"), function () {
|
frm.add_custom_button(__("Unlink external integrations"), function() {
|
||||||
frappe.confirm(
|
frappe.confirm(__("This action will unlink this account from any external service integrating ERPNext with your bank accounts. It cannot be undone. Are you certain ?"), function() {
|
||||||
__(
|
frm.set_value("integration_id", "");
|
||||||
"This action will unlink this account from any external service integrating ERPNext with your bank accounts. It cannot be undone. Are you certain ?"
|
});
|
||||||
),
|
|
||||||
function () {
|
|
||||||
frm.set_value("integration_id", "");
|
|
||||||
}
|
|
||||||
);
|
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
is_company_account: function (frm) {
|
is_company_account: function(frm) {
|
||||||
frm.set_df_property("account", "reqd", frm.doc.is_company_account);
|
frm.set_df_property('account', 'reqd', frm.doc.is_company_account);
|
||||||
},
|
}
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -9,7 +9,6 @@ from frappe.contacts.address_and_contact import (
|
|||||||
load_address_and_contact,
|
load_address_and_contact,
|
||||||
)
|
)
|
||||||
from frappe.model.document import Document
|
from frappe.model.document import Document
|
||||||
from frappe.utils import comma_and, get_link_to_form
|
|
||||||
|
|
||||||
|
|
||||||
class BankAccount(Document):
|
class BankAccount(Document):
|
||||||
@@ -53,19 +52,6 @@ class BankAccount(Document):
|
|||||||
def validate(self):
|
def validate(self):
|
||||||
self.validate_company()
|
self.validate_company()
|
||||||
self.validate_iban()
|
self.validate_iban()
|
||||||
self.validate_account()
|
|
||||||
|
|
||||||
def validate_account(self):
|
|
||||||
if self.account:
|
|
||||||
if accounts := frappe.db.get_all(
|
|
||||||
"Bank Account", filters={"account": self.account, "name": ["!=", self.name]}, as_list=1
|
|
||||||
):
|
|
||||||
frappe.throw(
|
|
||||||
_("'{0}' account is already used by {1}. Use another account.").format(
|
|
||||||
frappe.bold(self.account),
|
|
||||||
frappe.bold(comma_and([get_link_to_form(self.doctype, x[0]) for x in accounts])),
|
|
||||||
)
|
|
||||||
)
|
|
||||||
|
|
||||||
def validate_company(self):
|
def validate_company(self):
|
||||||
if self.is_company_account and not self.company:
|
if self.is_company_account and not self.company:
|
||||||
|
|||||||
@@ -1,6 +1,8 @@
|
|||||||
// Copyright (c) 2018, Frappe Technologies Pvt. Ltd. and contributors
|
// Copyright (c) 2018, Frappe Technologies Pvt. Ltd. and contributors
|
||||||
// For license information, please see license.txt
|
// For license information, please see license.txt
|
||||||
|
|
||||||
frappe.ui.form.on("Bank Account Subtype", {
|
frappe.ui.form.on('Bank Account Subtype', {
|
||||||
refresh: function () {},
|
refresh: function() {
|
||||||
|
|
||||||
|
}
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -1,7 +1,8 @@
|
|||||||
// Copyright (c) 2020, Frappe Technologies Pvt. Ltd. and contributors
|
// Copyright (c) 2020, Frappe Technologies Pvt. Ltd. and contributors
|
||||||
// For license information, please see license.txt
|
// For license information, please see license.txt
|
||||||
|
|
||||||
frappe.ui.form.on("Bank Account Type", {
|
frappe.ui.form.on('Bank Account Type', {
|
||||||
// refresh: function(frm) {
|
// refresh: function(frm) {
|
||||||
|
|
||||||
// }
|
// }
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -2,76 +2,80 @@
|
|||||||
// License: GNU General Public License v3. See license.txt
|
// License: GNU General Public License v3. See license.txt
|
||||||
|
|
||||||
frappe.ui.form.on("Bank Clearance", {
|
frappe.ui.form.on("Bank Clearance", {
|
||||||
setup: function (frm) {
|
setup: function(frm) {
|
||||||
frm.add_fetch("account", "account_currency", "account_currency");
|
frm.add_fetch("account", "account_currency", "account_currency");
|
||||||
|
|
||||||
frm.set_query("account", function () {
|
frm.set_query("account", function() {
|
||||||
return {
|
return {
|
||||||
filters: {
|
"filters": {
|
||||||
account_type: ["in", ["Bank", "Cash"]],
|
"account_type": ["in",["Bank","Cash"]],
|
||||||
is_group: 0,
|
"is_group": 0,
|
||||||
},
|
}
|
||||||
};
|
};
|
||||||
});
|
});
|
||||||
|
|
||||||
frm.set_query("bank_account", function () {
|
frm.set_query("bank_account", function () {
|
||||||
return {
|
return {
|
||||||
filters: {
|
filters: {
|
||||||
is_company_account: 1,
|
'is_company_account': 1
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
|
|
||||||
onload: function (frm) {
|
onload: function(frm) {
|
||||||
let default_bank_account = frappe.defaults.get_user_default("Company")
|
|
||||||
? locals[":Company"][frappe.defaults.get_user_default("Company")]["default_bank_account"]
|
let default_bank_account = frappe.defaults.get_user_default("Company")?
|
||||||
: "";
|
locals[":Company"][frappe.defaults.get_user_default("Company")]["default_bank_account"]: "";
|
||||||
frm.set_value("account", default_bank_account);
|
frm.set_value("account", default_bank_account);
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
frm.set_value("from_date", frappe.datetime.month_start());
|
frm.set_value("from_date", frappe.datetime.month_start());
|
||||||
frm.set_value("to_date", frappe.datetime.month_end());
|
frm.set_value("to_date", frappe.datetime.month_end());
|
||||||
},
|
},
|
||||||
|
|
||||||
refresh: function (frm) {
|
refresh: function(frm) {
|
||||||
frm.disable_save();
|
frm.disable_save();
|
||||||
frm.add_custom_button(__("Get Payment Entries"), () => frm.trigger("get_payment_entries"));
|
frm.add_custom_button(__('Get Payment Entries'), () =>
|
||||||
|
frm.trigger("get_payment_entries")
|
||||||
|
);
|
||||||
|
|
||||||
frm.change_custom_button_type(__("Get Payment Entries"), null, "primary");
|
frm.change_custom_button_type(__('Get Payment Entries'), null, 'primary');
|
||||||
},
|
},
|
||||||
|
|
||||||
update_clearance_date: function (frm) {
|
update_clearance_date: function(frm) {
|
||||||
return frappe.call({
|
return frappe.call({
|
||||||
method: "update_clearance_date",
|
method: "update_clearance_date",
|
||||||
doc: frm.doc,
|
doc: frm.doc,
|
||||||
callback: function (r, rt) {
|
callback: function(r, rt) {
|
||||||
frm.refresh_field("payment_entries");
|
frm.refresh_field("payment_entries");
|
||||||
frm.refresh_fields();
|
frm.refresh_fields();
|
||||||
|
|
||||||
if (!frm.doc.payment_entries.length) {
|
if (!frm.doc.payment_entries.length) {
|
||||||
frm.change_custom_button_type(__("Get Payment Entries"), null, "primary");
|
frm.change_custom_button_type(__('Get Payment Entries'), null, 'primary');
|
||||||
frm.change_custom_button_type(__("Update Clearance Date"), null, "default");
|
frm.change_custom_button_type(__('Update Clearance Date'), null, 'default');
|
||||||
}
|
}
|
||||||
},
|
}
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
|
|
||||||
get_payment_entries: function (frm) {
|
get_payment_entries: function(frm) {
|
||||||
return frappe.call({
|
return frappe.call({
|
||||||
method: "get_payment_entries",
|
method: "get_payment_entries",
|
||||||
doc: frm.doc,
|
doc: frm.doc,
|
||||||
callback: function (r, rt) {
|
callback: function(r, rt) {
|
||||||
frm.refresh_field("payment_entries");
|
frm.refresh_field("payment_entries");
|
||||||
|
|
||||||
if (frm.doc.payment_entries.length) {
|
if (frm.doc.payment_entries.length) {
|
||||||
frm.add_custom_button(__("Update Clearance Date"), () =>
|
frm.add_custom_button(__('Update Clearance Date'), () =>
|
||||||
frm.trigger("update_clearance_date")
|
frm.trigger("update_clearance_date")
|
||||||
);
|
);
|
||||||
|
|
||||||
frm.change_custom_button_type(__("Get Payment Entries"), null, "default");
|
frm.change_custom_button_type(__('Get Payment Entries'), null, 'default');
|
||||||
frm.change_custom_button_type(__("Update Clearance Date"), null, "primary");
|
frm.change_custom_button_type(__('Update Clearance Date'), null, 'primary');
|
||||||
}
|
}
|
||||||
},
|
}
|
||||||
});
|
});
|
||||||
},
|
}
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -5,9 +5,7 @@
|
|||||||
import frappe
|
import frappe
|
||||||
from frappe import _, msgprint
|
from frappe import _, msgprint
|
||||||
from frappe.model.document import Document
|
from frappe.model.document import Document
|
||||||
from frappe.query_builder.custom import ConstantColumn
|
|
||||||
from frappe.utils import flt, fmt_money, getdate
|
from frappe.utils import flt, fmt_money, getdate
|
||||||
from pypika import Order
|
|
||||||
|
|
||||||
import erpnext
|
import erpnext
|
||||||
|
|
||||||
@@ -181,62 +179,39 @@ def get_payment_entries_for_bank_clearance(
|
|||||||
|
|
||||||
pos_sales_invoices, pos_purchase_invoices = [], []
|
pos_sales_invoices, pos_purchase_invoices = [], []
|
||||||
if include_pos_transactions:
|
if include_pos_transactions:
|
||||||
si_payment = frappe.qb.DocType("Sales Invoice Payment")
|
pos_sales_invoices = frappe.db.sql(
|
||||||
si = frappe.qb.DocType("Sales Invoice")
|
"""
|
||||||
acc = frappe.qb.DocType("Account")
|
select
|
||||||
|
"Sales Invoice Payment" as payment_document, sip.name as payment_entry, sip.amount as debit,
|
||||||
|
si.posting_date, si.customer as against_account, sip.clearance_date,
|
||||||
|
account.account_currency, 0 as credit
|
||||||
|
from `tabSales Invoice Payment` sip, `tabSales Invoice` si, `tabAccount` account
|
||||||
|
where
|
||||||
|
sip.account=%(account)s and si.docstatus=1 and sip.parent = si.name
|
||||||
|
and account.name = sip.account and si.posting_date >= %(from)s and si.posting_date <= %(to)s
|
||||||
|
order by
|
||||||
|
si.posting_date ASC, si.name DESC
|
||||||
|
""",
|
||||||
|
{"account": account, "from": from_date, "to": to_date},
|
||||||
|
as_dict=1,
|
||||||
|
)
|
||||||
|
|
||||||
pos_sales_invoices = (
|
pos_purchase_invoices = frappe.db.sql(
|
||||||
frappe.qb.from_(si_payment)
|
"""
|
||||||
.inner_join(si)
|
select
|
||||||
.on(si_payment.parent == si.name)
|
"Purchase Invoice" as payment_document, pi.name as payment_entry, pi.paid_amount as credit,
|
||||||
.inner_join(acc)
|
pi.posting_date, pi.supplier as against_account, pi.clearance_date,
|
||||||
.on(si_payment.account == acc.name)
|
account.account_currency, 0 as debit
|
||||||
.select(
|
from `tabPurchase Invoice` pi, `tabAccount` account
|
||||||
ConstantColumn("Sales Invoice").as_("payment_document"),
|
where
|
||||||
si.name.as_("payment_entry"),
|
pi.cash_bank_account=%(account)s and pi.docstatus=1 and account.name = pi.cash_bank_account
|
||||||
si_payment.reference_no.as_("cheque_number"),
|
and pi.posting_date >= %(from)s and pi.posting_date <= %(to)s
|
||||||
si_payment.amount.as_("debit"),
|
order by
|
||||||
si.posting_date,
|
pi.posting_date ASC, pi.name DESC
|
||||||
si.customer.as_("against_account"),
|
""",
|
||||||
si_payment.clearance_date,
|
{"account": account, "from": from_date, "to": to_date},
|
||||||
acc.account_currency,
|
as_dict=1,
|
||||||
ConstantColumn(0).as_("credit"),
|
)
|
||||||
)
|
|
||||||
.where(
|
|
||||||
(si.docstatus == 1)
|
|
||||||
& (si_payment.account == account)
|
|
||||||
& (si.posting_date >= from_date)
|
|
||||||
& (si.posting_date <= to_date)
|
|
||||||
)
|
|
||||||
.orderby(si.posting_date)
|
|
||||||
.orderby(si.name, order=Order.desc)
|
|
||||||
).run(as_dict=True)
|
|
||||||
|
|
||||||
pi = frappe.qb.DocType("Purchase Invoice")
|
|
||||||
|
|
||||||
pos_purchase_invoices = (
|
|
||||||
frappe.qb.from_(pi)
|
|
||||||
.inner_join(acc)
|
|
||||||
.on(pi.cash_bank_account == acc.name)
|
|
||||||
.select(
|
|
||||||
ConstantColumn("Purchase Invoice").as_("payment_document"),
|
|
||||||
pi.name.as_("payment_entry"),
|
|
||||||
pi.paid_amount.as_("credit"),
|
|
||||||
pi.posting_date,
|
|
||||||
pi.supplier.as_("against_account"),
|
|
||||||
pi.clearance_date,
|
|
||||||
acc.account_currency,
|
|
||||||
ConstantColumn(0).as_("debit"),
|
|
||||||
)
|
|
||||||
.where(
|
|
||||||
(pi.docstatus == 1)
|
|
||||||
& (pi.cash_bank_account == account)
|
|
||||||
& (pi.posting_date >= from_date)
|
|
||||||
& (pi.posting_date <= to_date)
|
|
||||||
)
|
|
||||||
.orderby(pi.posting_date)
|
|
||||||
.orderby(pi.name, order=Order.desc)
|
|
||||||
).run(as_dict=True)
|
|
||||||
|
|
||||||
entries = (
|
entries = (
|
||||||
list(payment_entries)
|
list(payment_entries)
|
||||||
|
|||||||
@@ -1,39 +1,39 @@
|
|||||||
// Copyright (c) 2016, Frappe Technologies Pvt. Ltd. and contributors
|
// Copyright (c) 2016, Frappe Technologies Pvt. Ltd. and contributors
|
||||||
// For license information, please see license.txt
|
// For license information, please see license.txt
|
||||||
|
|
||||||
cur_frm.add_fetch("bank_account", "account", "account");
|
cur_frm.add_fetch('bank_account','account','account');
|
||||||
cur_frm.add_fetch("bank_account", "bank_account_no", "bank_account_no");
|
cur_frm.add_fetch('bank_account','bank_account_no','bank_account_no');
|
||||||
cur_frm.add_fetch("bank_account", "iban", "iban");
|
cur_frm.add_fetch('bank_account','iban','iban');
|
||||||
cur_frm.add_fetch("bank_account", "branch_code", "branch_code");
|
cur_frm.add_fetch('bank_account','branch_code','branch_code');
|
||||||
cur_frm.add_fetch("bank", "swift_number", "swift_number");
|
cur_frm.add_fetch('bank','swift_number','swift_number');
|
||||||
|
|
||||||
frappe.ui.form.on("Bank Guarantee", {
|
frappe.ui.form.on('Bank Guarantee', {
|
||||||
setup: function (frm) {
|
setup: function(frm) {
|
||||||
frm.set_query("bank", function () {
|
frm.set_query("bank", function() {
|
||||||
|
return {
|
||||||
|
filters: {
|
||||||
|
company: frm.doc.company
|
||||||
|
}
|
||||||
|
};
|
||||||
|
});
|
||||||
|
frm.set_query("bank_account", function() {
|
||||||
return {
|
return {
|
||||||
filters: {
|
filters: {
|
||||||
company: frm.doc.company,
|
company: frm.doc.company,
|
||||||
},
|
bank: frm.doc.bank
|
||||||
};
|
}
|
||||||
|
}
|
||||||
});
|
});
|
||||||
frm.set_query("bank_account", function () {
|
frm.set_query("project", function() {
|
||||||
return {
|
return {
|
||||||
filters: {
|
filters: {
|
||||||
company: frm.doc.company,
|
customer: frm.doc.customer
|
||||||
bank: frm.doc.bank,
|
}
|
||||||
},
|
|
||||||
};
|
|
||||||
});
|
|
||||||
frm.set_query("project", function () {
|
|
||||||
return {
|
|
||||||
filters: {
|
|
||||||
customer: frm.doc.customer,
|
|
||||||
},
|
|
||||||
};
|
};
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
|
|
||||||
bg_type: function (frm) {
|
bg_type: function(frm) {
|
||||||
if (frm.doc.bg_type == "Receiving") {
|
if (frm.doc.bg_type == "Receiving") {
|
||||||
frm.set_value("reference_doctype", "Sales Order");
|
frm.set_value("reference_doctype", "Sales Order");
|
||||||
} else if (frm.doc.bg_type == "Providing") {
|
} else if (frm.doc.bg_type == "Providing") {
|
||||||
@@ -41,33 +41,34 @@ frappe.ui.form.on("Bank Guarantee", {
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
reference_docname: function (frm) {
|
reference_docname: function(frm) {
|
||||||
if (frm.doc.reference_docname && frm.doc.reference_doctype) {
|
if (frm.doc.reference_docname && frm.doc.reference_doctype) {
|
||||||
let party_field = frm.doc.reference_doctype == "Sales Order" ? "customer" : "supplier";
|
let party_field = frm.doc.reference_doctype == "Sales Order" ? "customer" : "supplier";
|
||||||
|
|
||||||
frappe.call({
|
frappe.call({
|
||||||
method: "erpnext.accounts.doctype.bank_guarantee.bank_guarantee.get_voucher_details",
|
method: "erpnext.accounts.doctype.bank_guarantee.bank_guarantee.get_voucher_details",
|
||||||
args: {
|
args: {
|
||||||
bank_guarantee_type: frm.doc.bg_type,
|
"bank_guarantee_type": frm.doc.bg_type,
|
||||||
reference_name: frm.doc.reference_docname,
|
"reference_name": frm.doc.reference_docname
|
||||||
},
|
},
|
||||||
callback: function (r) {
|
callback: function(r) {
|
||||||
if (r.message) {
|
if (r.message) {
|
||||||
if (r.message[party_field]) frm.set_value(party_field, r.message[party_field]);
|
if (r.message[party_field]) frm.set_value(party_field, r.message[party_field]);
|
||||||
if (r.message.project) frm.set_value("project", r.message.project);
|
if (r.message.project) frm.set_value("project", r.message.project);
|
||||||
if (r.message.grand_total) frm.set_value("amount", r.message.grand_total);
|
if (r.message.grand_total) frm.set_value("amount", r.message.grand_total);
|
||||||
}
|
}
|
||||||
},
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
start_date: function (frm) {
|
start_date: function(frm) {
|
||||||
var end_date = frappe.datetime.add_days(cur_frm.doc.start_date, cur_frm.doc.validity - 1);
|
var end_date = frappe.datetime.add_days(cur_frm.doc.start_date, cur_frm.doc.validity - 1);
|
||||||
cur_frm.set_value("end_date", end_date);
|
cur_frm.set_value("end_date", end_date);
|
||||||
},
|
},
|
||||||
validity: function (frm) {
|
validity: function(frm) {
|
||||||
var end_date = frappe.datetime.add_days(cur_frm.doc.start_date, cur_frm.doc.validity - 1);
|
var end_date = frappe.datetime.add_days(cur_frm.doc.start_date, cur_frm.doc.validity - 1);
|
||||||
cur_frm.set_value("end_date", end_date);
|
cur_frm.set_value("end_date", end_date);
|
||||||
},
|
}
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -8,22 +8,21 @@ frappe.ui.form.on("Bank Reconciliation Tool", {
|
|||||||
return {
|
return {
|
||||||
filters: {
|
filters: {
|
||||||
company: frm.doc.company,
|
company: frm.doc.company,
|
||||||
is_company_account: 1,
|
'is_company_account': 1
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
});
|
});
|
||||||
let no_bank_transactions_text = `<div class="text-muted text-center">${__(
|
let no_bank_transactions_text =
|
||||||
"No Matching Bank Transactions Found"
|
`<div class="text-muted text-center">${__("No Matching Bank Transactions Found")}</div>`
|
||||||
)}</div>`;
|
|
||||||
set_field_options("no_bank_transactions", no_bank_transactions_text);
|
set_field_options("no_bank_transactions", no_bank_transactions_text);
|
||||||
},
|
},
|
||||||
|
|
||||||
onload: function (frm) {
|
onload: function (frm) {
|
||||||
// Set default filter dates
|
// Set default filter dates
|
||||||
let today = frappe.datetime.get_today();
|
let today = frappe.datetime.get_today()
|
||||||
frm.doc.bank_statement_from_date = frappe.datetime.add_months(today, -1);
|
frm.doc.bank_statement_from_date = frappe.datetime.add_months(today, -1);
|
||||||
frm.doc.bank_statement_to_date = today;
|
frm.doc.bank_statement_to_date = today;
|
||||||
frm.trigger("bank_account");
|
frm.trigger('bank_account');
|
||||||
},
|
},
|
||||||
|
|
||||||
filter_by_reference_date: function (frm) {
|
filter_by_reference_date: function (frm) {
|
||||||
@@ -38,27 +37,34 @@ frappe.ui.form.on("Bank Reconciliation Tool", {
|
|||||||
|
|
||||||
refresh: function (frm) {
|
refresh: function (frm) {
|
||||||
frm.disable_save();
|
frm.disable_save();
|
||||||
frappe.require("bank-reconciliation-tool.bundle.js", () => frm.trigger("make_reconciliation_tool"));
|
frappe.require("bank-reconciliation-tool.bundle.js", () =>
|
||||||
|
frm.trigger("make_reconciliation_tool")
|
||||||
frm.add_custom_button(__("Upload Bank Statement"), () =>
|
|
||||||
frappe.call({
|
|
||||||
method: "erpnext.accounts.doctype.bank_statement_import.bank_statement_import.upload_bank_statement",
|
|
||||||
args: {
|
|
||||||
dt: frm.doc.doctype,
|
|
||||||
dn: frm.doc.name,
|
|
||||||
company: frm.doc.company,
|
|
||||||
bank_account: frm.doc.bank_account,
|
|
||||||
},
|
|
||||||
callback: function (r) {
|
|
||||||
if (!r.exc) {
|
|
||||||
var doc = frappe.model.sync(r.message);
|
|
||||||
frappe.set_route("Form", doc[0].doctype, doc[0].name);
|
|
||||||
}
|
|
||||||
},
|
|
||||||
})
|
|
||||||
);
|
);
|
||||||
|
|
||||||
frm.add_custom_button(__("Auto Reconcile"), function () {
|
frm.add_custom_button(__("Upload Bank Statement"), () =>
|
||||||
|
frappe.call({
|
||||||
|
method:
|
||||||
|
"erpnext.accounts.doctype.bank_statement_import.bank_statement_import.upload_bank_statement",
|
||||||
|
args: {
|
||||||
|
dt: frm.doc.doctype,
|
||||||
|
dn: frm.doc.name,
|
||||||
|
company: frm.doc.company,
|
||||||
|
bank_account: frm.doc.bank_account,
|
||||||
|
},
|
||||||
|
callback: function (r) {
|
||||||
|
if (!r.exc) {
|
||||||
|
var doc = frappe.model.sync(r.message);
|
||||||
|
frappe.set_route(
|
||||||
|
"Form",
|
||||||
|
doc[0].doctype,
|
||||||
|
doc[0].name
|
||||||
|
);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
})
|
||||||
|
);
|
||||||
|
|
||||||
|
frm.add_custom_button(__('Auto Reconcile'), function() {
|
||||||
frappe.call({
|
frappe.call({
|
||||||
method: "erpnext.accounts.doctype.bank_reconciliation_tool.bank_reconciliation_tool.auto_reconcile_vouchers",
|
method: "erpnext.accounts.doctype.bank_reconciliation_tool.bank_reconciliation_tool.auto_reconcile_vouchers",
|
||||||
args: {
|
args: {
|
||||||
@@ -69,22 +75,33 @@ frappe.ui.form.on("Bank Reconciliation Tool", {
|
|||||||
from_reference_date: frm.doc.from_reference_date,
|
from_reference_date: frm.doc.from_reference_date,
|
||||||
to_reference_date: frm.doc.to_reference_date,
|
to_reference_date: frm.doc.to_reference_date,
|
||||||
},
|
},
|
||||||
});
|
})
|
||||||
});
|
});
|
||||||
|
|
||||||
frm.add_custom_button(__("Get Unreconciled Entries"), function () {
|
frm.add_custom_button(__('Get Unreconciled Entries'), function() {
|
||||||
frm.trigger("make_reconciliation_tool");
|
frm.trigger("make_reconciliation_tool");
|
||||||
});
|
});
|
||||||
frm.change_custom_button_type(__("Get Unreconciled Entries"), null, "primary");
|
frm.change_custom_button_type(__('Get Unreconciled Entries'), null, 'primary');
|
||||||
|
|
||||||
},
|
},
|
||||||
|
|
||||||
bank_account: function (frm) {
|
bank_account: function (frm) {
|
||||||
frappe.db.get_value("Bank Account", frm.doc.bank_account, "account", (r) => {
|
frappe.db.get_value(
|
||||||
frappe.db.get_value("Account", r.account, "account_currency", (r) => {
|
"Bank Account",
|
||||||
frm.doc.account_currency = r.account_currency;
|
frm.doc.bank_account,
|
||||||
frm.trigger("render_chart");
|
"account",
|
||||||
});
|
(r) => {
|
||||||
});
|
frappe.db.get_value(
|
||||||
|
"Account",
|
||||||
|
r.account,
|
||||||
|
"account_currency",
|
||||||
|
(r) => {
|
||||||
|
frm.doc.account_currency = r.account_currency;
|
||||||
|
frm.trigger("render_chart");
|
||||||
|
}
|
||||||
|
);
|
||||||
|
}
|
||||||
|
);
|
||||||
frm.trigger("get_account_opening_balance");
|
frm.trigger("get_account_opening_balance");
|
||||||
},
|
},
|
||||||
|
|
||||||
@@ -103,7 +120,11 @@ frappe.ui.form.on("Bank Reconciliation Tool", {
|
|||||||
) {
|
) {
|
||||||
frm.trigger("render_chart");
|
frm.trigger("render_chart");
|
||||||
frm.trigger("render");
|
frm.trigger("render");
|
||||||
frappe.utils.scroll_to(frm.get_field("reconciliation_tool_cards").$wrapper, true, 30);
|
frappe.utils.scroll_to(
|
||||||
|
frm.get_field("reconciliation_tool_cards").$wrapper,
|
||||||
|
true,
|
||||||
|
30
|
||||||
|
);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
@@ -112,10 +133,11 @@ frappe.ui.form.on("Bank Reconciliation Tool", {
|
|||||||
get_account_opening_balance(frm) {
|
get_account_opening_balance(frm) {
|
||||||
if (frm.doc.bank_account && frm.doc.bank_statement_from_date) {
|
if (frm.doc.bank_account && frm.doc.bank_statement_from_date) {
|
||||||
frappe.call({
|
frappe.call({
|
||||||
method: "erpnext.accounts.doctype.bank_reconciliation_tool.bank_reconciliation_tool.get_account_balance",
|
method:
|
||||||
|
"erpnext.accounts.doctype.bank_reconciliation_tool.bank_reconciliation_tool.get_account_balance",
|
||||||
args: {
|
args: {
|
||||||
bank_account: frm.doc.bank_account,
|
bank_account: frm.doc.bank_account,
|
||||||
till_date: frappe.datetime.add_days(frm.doc.bank_statement_from_date, -1),
|
till_date: frappe.datetime.add_days(frm.doc.bank_statement_from_date, -1)
|
||||||
},
|
},
|
||||||
callback: (response) => {
|
callback: (response) => {
|
||||||
frm.set_value("account_opening_balance", response.message);
|
frm.set_value("account_opening_balance", response.message);
|
||||||
@@ -127,7 +149,8 @@ frappe.ui.form.on("Bank Reconciliation Tool", {
|
|||||||
get_cleared_balance(frm) {
|
get_cleared_balance(frm) {
|
||||||
if (frm.doc.bank_account && frm.doc.bank_statement_to_date) {
|
if (frm.doc.bank_account && frm.doc.bank_statement_to_date) {
|
||||||
return frappe.call({
|
return frappe.call({
|
||||||
method: "erpnext.accounts.doctype.bank_reconciliation_tool.bank_reconciliation_tool.get_account_balance",
|
method:
|
||||||
|
"erpnext.accounts.doctype.bank_reconciliation_tool.bank_reconciliation_tool.get_account_balance",
|
||||||
args: {
|
args: {
|
||||||
bank_account: frm.doc.bank_account,
|
bank_account: frm.doc.bank_account,
|
||||||
till_date: frm.doc.bank_statement_to_date,
|
till_date: frm.doc.bank_statement_to_date,
|
||||||
@@ -140,30 +163,41 @@ frappe.ui.form.on("Bank Reconciliation Tool", {
|
|||||||
},
|
},
|
||||||
|
|
||||||
render_chart(frm) {
|
render_chart(frm) {
|
||||||
frm.cards_manager = new erpnext.accounts.bank_reconciliation.NumberCardManager({
|
frm.cards_manager = new erpnext.accounts.bank_reconciliation.NumberCardManager(
|
||||||
$reconciliation_tool_cards: frm.get_field("reconciliation_tool_cards").$wrapper,
|
{
|
||||||
bank_statement_closing_balance: frm.doc.bank_statement_closing_balance,
|
$reconciliation_tool_cards: frm.get_field(
|
||||||
cleared_balance: frm.cleared_balance,
|
"reconciliation_tool_cards"
|
||||||
currency: frm.doc.account_currency,
|
).$wrapper,
|
||||||
});
|
bank_statement_closing_balance:
|
||||||
|
frm.doc.bank_statement_closing_balance,
|
||||||
|
cleared_balance: frm.cleared_balance,
|
||||||
|
currency: frm.doc.account_currency,
|
||||||
|
}
|
||||||
|
);
|
||||||
},
|
},
|
||||||
|
|
||||||
render(frm) {
|
render(frm) {
|
||||||
if (frm.doc.bank_account) {
|
if (frm.doc.bank_account) {
|
||||||
frm.bank_reconciliation_data_table_manager =
|
frm.bank_reconciliation_data_table_manager = new erpnext.accounts.bank_reconciliation.DataTableManager(
|
||||||
new erpnext.accounts.bank_reconciliation.DataTableManager({
|
{
|
||||||
company: frm.doc.company,
|
company: frm.doc.company,
|
||||||
bank_account: frm.doc.bank_account,
|
bank_account: frm.doc.bank_account,
|
||||||
$reconciliation_tool_dt: frm.get_field("reconciliation_tool_dt").$wrapper,
|
$reconciliation_tool_dt: frm.get_field(
|
||||||
$no_bank_transactions: frm.get_field("no_bank_transactions").$wrapper,
|
"reconciliation_tool_dt"
|
||||||
|
).$wrapper,
|
||||||
|
$no_bank_transactions: frm.get_field(
|
||||||
|
"no_bank_transactions"
|
||||||
|
).$wrapper,
|
||||||
bank_statement_from_date: frm.doc.bank_statement_from_date,
|
bank_statement_from_date: frm.doc.bank_statement_from_date,
|
||||||
bank_statement_to_date: frm.doc.bank_statement_to_date,
|
bank_statement_to_date: frm.doc.bank_statement_to_date,
|
||||||
filter_by_reference_date: frm.doc.filter_by_reference_date,
|
filter_by_reference_date: frm.doc.filter_by_reference_date,
|
||||||
from_reference_date: frm.doc.from_reference_date,
|
from_reference_date: frm.doc.from_reference_date,
|
||||||
to_reference_date: frm.doc.to_reference_date,
|
to_reference_date: frm.doc.to_reference_date,
|
||||||
bank_statement_closing_balance: frm.doc.bank_statement_closing_balance,
|
bank_statement_closing_balance:
|
||||||
|
frm.doc.bank_statement_closing_balance,
|
||||||
cards_manager: frm.cards_manager,
|
cards_manager: frm.cards_manager,
|
||||||
});
|
}
|
||||||
|
);
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -9,6 +9,7 @@ from frappe import _
|
|||||||
from frappe.model.document import Document
|
from frappe.model.document import Document
|
||||||
from frappe.query_builder.custom import ConstantColumn
|
from frappe.query_builder.custom import ConstantColumn
|
||||||
from frappe.utils import cint, flt
|
from frappe.utils import cint, flt
|
||||||
|
from pypika.terms import Parameter
|
||||||
|
|
||||||
from erpnext import get_default_cost_center
|
from erpnext import get_default_cost_center
|
||||||
from erpnext.accounts.doctype.bank_transaction.bank_transaction import get_total_allocated_amount
|
from erpnext.accounts.doctype.bank_transaction.bank_transaction import get_total_allocated_amount
|
||||||
@@ -508,18 +509,6 @@ def check_matching(
|
|||||||
to_reference_date,
|
to_reference_date,
|
||||||
):
|
):
|
||||||
exact_match = True if "exact_match" in document_types else False
|
exact_match = True if "exact_match" in document_types else False
|
||||||
|
|
||||||
common_filters = frappe._dict(
|
|
||||||
{
|
|
||||||
"amount": transaction.unallocated_amount,
|
|
||||||
"payment_type": "Receive" if transaction.deposit > 0.0 else "Pay",
|
|
||||||
"reference_no": transaction.reference_number,
|
|
||||||
"party_type": transaction.party_type,
|
|
||||||
"party": transaction.party,
|
|
||||||
"bank_account": bank_account,
|
|
||||||
}
|
|
||||||
)
|
|
||||||
|
|
||||||
queries = get_queries(
|
queries = get_queries(
|
||||||
bank_account,
|
bank_account,
|
||||||
company,
|
company,
|
||||||
@@ -531,12 +520,20 @@ def check_matching(
|
|||||||
from_reference_date,
|
from_reference_date,
|
||||||
to_reference_date,
|
to_reference_date,
|
||||||
exact_match,
|
exact_match,
|
||||||
common_filters,
|
|
||||||
)
|
)
|
||||||
|
|
||||||
|
filters = {
|
||||||
|
"amount": transaction.unallocated_amount,
|
||||||
|
"payment_type": "Receive" if transaction.deposit > 0.0 else "Pay",
|
||||||
|
"reference_no": transaction.reference_number,
|
||||||
|
"party_type": transaction.party_type,
|
||||||
|
"party": transaction.party,
|
||||||
|
"bank_account": bank_account,
|
||||||
|
}
|
||||||
|
|
||||||
matching_vouchers = []
|
matching_vouchers = []
|
||||||
for query in queries:
|
for query in queries:
|
||||||
matching_vouchers.extend(query.run(as_dict=True))
|
matching_vouchers.extend(frappe.db.sql(query, filters, as_dict=True))
|
||||||
|
|
||||||
return (
|
return (
|
||||||
sorted(matching_vouchers, key=lambda x: x["rank"], reverse=True) if matching_vouchers else []
|
sorted(matching_vouchers, key=lambda x: x["rank"], reverse=True) if matching_vouchers else []
|
||||||
@@ -554,7 +551,6 @@ def get_queries(
|
|||||||
from_reference_date,
|
from_reference_date,
|
||||||
to_reference_date,
|
to_reference_date,
|
||||||
exact_match,
|
exact_match,
|
||||||
common_filters,
|
|
||||||
):
|
):
|
||||||
# get queries to get matching vouchers
|
# get queries to get matching vouchers
|
||||||
account_from_to = "paid_to" if transaction.deposit > 0.0 else "paid_from"
|
account_from_to = "paid_to" if transaction.deposit > 0.0 else "paid_from"
|
||||||
@@ -575,7 +571,6 @@ def get_queries(
|
|||||||
filter_by_reference_date,
|
filter_by_reference_date,
|
||||||
from_reference_date,
|
from_reference_date,
|
||||||
to_reference_date,
|
to_reference_date,
|
||||||
common_filters,
|
|
||||||
)
|
)
|
||||||
or []
|
or []
|
||||||
)
|
)
|
||||||
@@ -595,7 +590,6 @@ def get_matching_queries(
|
|||||||
filter_by_reference_date,
|
filter_by_reference_date,
|
||||||
from_reference_date,
|
from_reference_date,
|
||||||
to_reference_date,
|
to_reference_date,
|
||||||
common_filters,
|
|
||||||
):
|
):
|
||||||
queries = []
|
queries = []
|
||||||
currency = get_account_currency(bank_account)
|
currency = get_account_currency(bank_account)
|
||||||
@@ -610,7 +604,6 @@ def get_matching_queries(
|
|||||||
filter_by_reference_date,
|
filter_by_reference_date,
|
||||||
from_reference_date,
|
from_reference_date,
|
||||||
to_reference_date,
|
to_reference_date,
|
||||||
common_filters,
|
|
||||||
)
|
)
|
||||||
queries.append(query)
|
queries.append(query)
|
||||||
|
|
||||||
@@ -623,17 +616,16 @@ def get_matching_queries(
|
|||||||
filter_by_reference_date,
|
filter_by_reference_date,
|
||||||
from_reference_date,
|
from_reference_date,
|
||||||
to_reference_date,
|
to_reference_date,
|
||||||
common_filters,
|
|
||||||
)
|
)
|
||||||
queries.append(query)
|
queries.append(query)
|
||||||
|
|
||||||
if transaction.deposit > 0.0 and "sales_invoice" in document_types:
|
if transaction.deposit > 0.0 and "sales_invoice" in document_types:
|
||||||
query = get_si_matching_query(exact_match, currency, common_filters)
|
query = get_si_matching_query(exact_match, currency)
|
||||||
queries.append(query)
|
queries.append(query)
|
||||||
|
|
||||||
if transaction.withdrawal > 0.0:
|
if transaction.withdrawal > 0.0:
|
||||||
if "purchase_invoice" in document_types:
|
if "purchase_invoice" in document_types:
|
||||||
query = get_pi_matching_query(exact_match, currency, common_filters)
|
query = get_pi_matching_query(exact_match, currency)
|
||||||
queries.append(query)
|
queries.append(query)
|
||||||
|
|
||||||
if "bank_transaction" in document_types:
|
if "bank_transaction" in document_types:
|
||||||
@@ -688,7 +680,7 @@ def get_bt_matching_query(exact_match, transaction):
|
|||||||
.where(amount_condition)
|
.where(amount_condition)
|
||||||
.where(bt.docstatus == 1)
|
.where(bt.docstatus == 1)
|
||||||
)
|
)
|
||||||
return query
|
return str(query)
|
||||||
|
|
||||||
|
|
||||||
def get_pe_matching_query(
|
def get_pe_matching_query(
|
||||||
@@ -700,7 +692,6 @@ def get_pe_matching_query(
|
|||||||
filter_by_reference_date,
|
filter_by_reference_date,
|
||||||
from_reference_date,
|
from_reference_date,
|
||||||
to_reference_date,
|
to_reference_date,
|
||||||
common_filters,
|
|
||||||
):
|
):
|
||||||
# get matching payment entries query
|
# get matching payment entries query
|
||||||
to_from = "to" if transaction.deposit > 0.0 else "from"
|
to_from = "to" if transaction.deposit > 0.0 else "from"
|
||||||
@@ -743,7 +734,7 @@ def get_pe_matching_query(
|
|||||||
.where(pe.docstatus == 1)
|
.where(pe.docstatus == 1)
|
||||||
.where(pe.payment_type.isin([payment_type, "Internal Transfer"]))
|
.where(pe.payment_type.isin([payment_type, "Internal Transfer"]))
|
||||||
.where(pe.clearance_date.isnull())
|
.where(pe.clearance_date.isnull())
|
||||||
.where(getattr(pe, account_from_to) == common_filters.bank_account)
|
.where(getattr(pe, account_from_to) == Parameter("%(bank_account)s"))
|
||||||
.where(amount_condition)
|
.where(amount_condition)
|
||||||
.where(filter_by_date)
|
.where(filter_by_date)
|
||||||
.orderby(pe.reference_date if cint(filter_by_reference_date) else pe.posting_date)
|
.orderby(pe.reference_date if cint(filter_by_reference_date) else pe.posting_date)
|
||||||
@@ -752,7 +743,7 @@ def get_pe_matching_query(
|
|||||||
if frappe.flags.auto_reconcile_vouchers == True:
|
if frappe.flags.auto_reconcile_vouchers == True:
|
||||||
query = query.where(ref_condition)
|
query = query.where(ref_condition)
|
||||||
|
|
||||||
return query
|
return str(query)
|
||||||
|
|
||||||
|
|
||||||
def get_je_matching_query(
|
def get_je_matching_query(
|
||||||
@@ -763,7 +754,6 @@ def get_je_matching_query(
|
|||||||
filter_by_reference_date,
|
filter_by_reference_date,
|
||||||
from_reference_date,
|
from_reference_date,
|
||||||
to_reference_date,
|
to_reference_date,
|
||||||
common_filters,
|
|
||||||
):
|
):
|
||||||
# get matching journal entry query
|
# get matching journal entry query
|
||||||
# We have mapping at the bank level
|
# We have mapping at the bank level
|
||||||
@@ -803,7 +793,7 @@ def get_je_matching_query(
|
|||||||
.where(je.docstatus == 1)
|
.where(je.docstatus == 1)
|
||||||
.where(je.voucher_type != "Opening Entry")
|
.where(je.voucher_type != "Opening Entry")
|
||||||
.where(je.clearance_date.isnull())
|
.where(je.clearance_date.isnull())
|
||||||
.where(jea.account == common_filters.bank_account)
|
.where(jea.account == Parameter("%(bank_account)s"))
|
||||||
.where(amount_equality if exact_match else getattr(jea, amount_field) > 0.0)
|
.where(amount_equality if exact_match else getattr(jea, amount_field) > 0.0)
|
||||||
.where(je.docstatus == 1)
|
.where(je.docstatus == 1)
|
||||||
.where(filter_by_date)
|
.where(filter_by_date)
|
||||||
@@ -813,19 +803,19 @@ def get_je_matching_query(
|
|||||||
if frappe.flags.auto_reconcile_vouchers == True:
|
if frappe.flags.auto_reconcile_vouchers == True:
|
||||||
query = query.where(ref_condition)
|
query = query.where(ref_condition)
|
||||||
|
|
||||||
return query
|
return str(query)
|
||||||
|
|
||||||
|
|
||||||
def get_si_matching_query(exact_match, currency, common_filters):
|
def get_si_matching_query(exact_match, currency):
|
||||||
# get matching sales invoice query
|
# get matching sales invoice query
|
||||||
si = frappe.qb.DocType("Sales Invoice")
|
si = frappe.qb.DocType("Sales Invoice")
|
||||||
sip = frappe.qb.DocType("Sales Invoice Payment")
|
sip = frappe.qb.DocType("Sales Invoice Payment")
|
||||||
|
|
||||||
amount_equality = sip.amount == common_filters.amount
|
amount_equality = sip.amount == Parameter("%(amount)s")
|
||||||
amount_rank = frappe.qb.terms.Case().when(amount_equality, 1).else_(0)
|
amount_rank = frappe.qb.terms.Case().when(amount_equality, 1).else_(0)
|
||||||
amount_condition = amount_equality if exact_match else sip.amount > 0.0
|
amount_condition = amount_equality if exact_match else sip.amount > 0.0
|
||||||
|
|
||||||
party_condition = si.customer == common_filters.party
|
party_condition = si.customer == Parameter("%(party)s")
|
||||||
party_rank = frappe.qb.terms.Case().when(party_condition, 1).else_(0)
|
party_rank = frappe.qb.terms.Case().when(party_condition, 1).else_(0)
|
||||||
|
|
||||||
query = (
|
query = (
|
||||||
@@ -846,23 +836,23 @@ def get_si_matching_query(exact_match, currency, common_filters):
|
|||||||
)
|
)
|
||||||
.where(si.docstatus == 1)
|
.where(si.docstatus == 1)
|
||||||
.where(sip.clearance_date.isnull())
|
.where(sip.clearance_date.isnull())
|
||||||
.where(sip.account == common_filters.bank_account)
|
.where(sip.account == Parameter("%(bank_account)s"))
|
||||||
.where(amount_condition)
|
.where(amount_condition)
|
||||||
.where(si.currency == currency)
|
.where(si.currency == currency)
|
||||||
)
|
)
|
||||||
|
|
||||||
return query
|
return str(query)
|
||||||
|
|
||||||
|
|
||||||
def get_pi_matching_query(exact_match, currency, common_filters):
|
def get_pi_matching_query(exact_match, currency):
|
||||||
# get matching purchase invoice query when they are also used as payment entries (is_paid)
|
# get matching purchase invoice query when they are also used as payment entries (is_paid)
|
||||||
purchase_invoice = frappe.qb.DocType("Purchase Invoice")
|
purchase_invoice = frappe.qb.DocType("Purchase Invoice")
|
||||||
|
|
||||||
amount_equality = purchase_invoice.paid_amount == common_filters.amount
|
amount_equality = purchase_invoice.paid_amount == Parameter("%(amount)s")
|
||||||
amount_rank = frappe.qb.terms.Case().when(amount_equality, 1).else_(0)
|
amount_rank = frappe.qb.terms.Case().when(amount_equality, 1).else_(0)
|
||||||
amount_condition = amount_equality if exact_match else purchase_invoice.paid_amount > 0.0
|
amount_condition = amount_equality if exact_match else purchase_invoice.paid_amount > 0.0
|
||||||
|
|
||||||
party_condition = purchase_invoice.supplier == common_filters.party
|
party_condition = purchase_invoice.supplier == Parameter("%(party)s")
|
||||||
party_rank = frappe.qb.terms.Case().when(party_condition, 1).else_(0)
|
party_rank = frappe.qb.terms.Case().when(party_condition, 1).else_(0)
|
||||||
|
|
||||||
query = (
|
query = (
|
||||||
@@ -882,9 +872,9 @@ def get_pi_matching_query(exact_match, currency, common_filters):
|
|||||||
.where(purchase_invoice.docstatus == 1)
|
.where(purchase_invoice.docstatus == 1)
|
||||||
.where(purchase_invoice.is_paid == 1)
|
.where(purchase_invoice.is_paid == 1)
|
||||||
.where(purchase_invoice.clearance_date.isnull())
|
.where(purchase_invoice.clearance_date.isnull())
|
||||||
.where(purchase_invoice.cash_bank_account == common_filters.bank_account)
|
.where(purchase_invoice.cash_bank_account == Parameter("%(bank_account)s"))
|
||||||
.where(amount_condition)
|
.where(amount_condition)
|
||||||
.where(purchase_invoice.currency == currency)
|
.where(purchase_invoice.currency == currency)
|
||||||
)
|
)
|
||||||
|
|
||||||
return query
|
return str(query)
|
||||||
|
|||||||
@@ -76,7 +76,6 @@ class TestBankReconciliationTool(AccountsTestMixin, FrappeTestCase):
|
|||||||
"deposit": 100,
|
"deposit": 100,
|
||||||
"bank_account": self.bank_account,
|
"bank_account": self.bank_account,
|
||||||
"reference_number": "123",
|
"reference_number": "123",
|
||||||
"currency": "INR",
|
|
||||||
}
|
}
|
||||||
)
|
)
|
||||||
.save()
|
.save()
|
||||||
|
|||||||
@@ -2,14 +2,26 @@
|
|||||||
// For license information, please see license.txt
|
// For license information, please see license.txt
|
||||||
|
|
||||||
frappe.ui.form.on("Bank Statement Import", {
|
frappe.ui.form.on("Bank Statement Import", {
|
||||||
|
onload(frm) {
|
||||||
|
frm.set_query("bank_account", function (doc) {
|
||||||
|
return {
|
||||||
|
filters: {
|
||||||
|
company: doc.company,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
});
|
||||||
|
},
|
||||||
|
|
||||||
setup(frm) {
|
setup(frm) {
|
||||||
frappe.realtime.on("data_import_refresh", ({ data_import }) => {
|
frappe.realtime.on("data_import_refresh", ({ data_import }) => {
|
||||||
frm.import_in_progress = false;
|
frm.import_in_progress = false;
|
||||||
if (data_import !== frm.doc.name) return;
|
if (data_import !== frm.doc.name) return;
|
||||||
frappe.model.clear_doc("Bank Statement Import", frm.doc.name);
|
frappe.model.clear_doc("Bank Statement Import", frm.doc.name);
|
||||||
frappe.model.with_doc("Bank Statement Import", frm.doc.name).then(() => {
|
frappe.model
|
||||||
frm.refresh();
|
.with_doc("Bank Statement Import", frm.doc.name)
|
||||||
});
|
.then(() => {
|
||||||
|
frm.refresh();
|
||||||
|
});
|
||||||
});
|
});
|
||||||
frappe.realtime.on("data_import_progress", (data) => {
|
frappe.realtime.on("data_import_progress", (data) => {
|
||||||
frm.import_in_progress = true;
|
frm.import_in_progress = true;
|
||||||
@@ -36,9 +48,20 @@ frappe.ui.form.on("Bank Statement Import", {
|
|||||||
: __("Updating {0} of {1}, {2}", message_args);
|
: __("Updating {0} of {1}, {2}", message_args);
|
||||||
}
|
}
|
||||||
if (data.skipping) {
|
if (data.skipping) {
|
||||||
message = __("Skipping {0} of {1}, {2}", [data.current, data.total, eta_message]);
|
message = __(
|
||||||
|
"Skipping {0} of {1}, {2}",
|
||||||
|
[
|
||||||
|
data.current,
|
||||||
|
data.total,
|
||||||
|
eta_message,
|
||||||
|
]
|
||||||
|
);
|
||||||
}
|
}
|
||||||
frm.dashboard.show_progress(__("Import Progress"), percent, message);
|
frm.dashboard.show_progress(
|
||||||
|
__("Import Progress"),
|
||||||
|
percent,
|
||||||
|
message
|
||||||
|
);
|
||||||
frm.page.set_indicator(__("In Progress"), "orange");
|
frm.page.set_indicator(__("In Progress"), "orange");
|
||||||
|
|
||||||
// hide progress when complete
|
// hide progress when complete
|
||||||
@@ -80,12 +103,15 @@ frappe.ui.form.on("Bank Statement Import", {
|
|||||||
frm.trigger("show_report_error_button");
|
frm.trigger("show_report_error_button");
|
||||||
|
|
||||||
if (frm.doc.status === "Partial Success") {
|
if (frm.doc.status === "Partial Success") {
|
||||||
frm.add_custom_button(__("Export Errored Rows"), () => frm.trigger("export_errored_rows"));
|
frm.add_custom_button(__("Export Errored Rows"), () =>
|
||||||
|
frm.trigger("export_errored_rows")
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (frm.doc.status.includes("Success")) {
|
if (frm.doc.status.includes("Success")) {
|
||||||
frm.add_custom_button(__("Go to {0} List", [__(frm.doc.reference_doctype)]), () =>
|
frm.add_custom_button(
|
||||||
frappe.set_route("List", frm.doc.reference_doctype)
|
__("Go to {0} List", [__(frm.doc.reference_doctype)]),
|
||||||
|
() => frappe.set_route("List", frm.doc.reference_doctype)
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
@@ -102,8 +128,13 @@ frappe.ui.form.on("Bank Statement Import", {
|
|||||||
frm.disable_save();
|
frm.disable_save();
|
||||||
if (frm.doc.status !== "Success") {
|
if (frm.doc.status !== "Success") {
|
||||||
if (!frm.is_new() && frm.has_import_file()) {
|
if (!frm.is_new() && frm.has_import_file()) {
|
||||||
let label = frm.doc.status === "Pending" ? __("Start Import") : __("Retry");
|
let label =
|
||||||
frm.page.set_primary_action(label, () => frm.events.start_import(frm));
|
frm.doc.status === "Pending"
|
||||||
|
? __("Start Import")
|
||||||
|
: __("Retry");
|
||||||
|
frm.page.set_primary_action(label, () =>
|
||||||
|
frm.events.start_import(frm)
|
||||||
|
);
|
||||||
} else {
|
} else {
|
||||||
frm.page.set_primary_action(__("Save"), () => frm.save());
|
frm.page.set_primary_action(__("Save"), () => frm.save());
|
||||||
}
|
}
|
||||||
@@ -145,24 +176,24 @@ frappe.ui.form.on("Bank Statement Import", {
|
|||||||
message =
|
message =
|
||||||
successful_records.length > 1
|
successful_records.length > 1
|
||||||
? __(
|
? __(
|
||||||
"Successfully imported {0} records out of {1}. Click on Export Errored Rows, fix the errors and import again.",
|
"Successfully imported {0} records out of {1}. Click on Export Errored Rows, fix the errors and import again.",
|
||||||
message_args
|
message_args
|
||||||
)
|
)
|
||||||
: __(
|
: __(
|
||||||
"Successfully imported {0} record out of {1}. Click on Export Errored Rows, fix the errors and import again.",
|
"Successfully imported {0} record out of {1}. Click on Export Errored Rows, fix the errors and import again.",
|
||||||
message_args
|
message_args
|
||||||
);
|
);
|
||||||
} else {
|
} else {
|
||||||
message =
|
message =
|
||||||
successful_records.length > 1
|
successful_records.length > 1
|
||||||
? __(
|
? __(
|
||||||
"Successfully updated {0} records out of {1}. Click on Export Errored Rows, fix the errors and import again.",
|
"Successfully updated {0} records out of {1}. Click on Export Errored Rows, fix the errors and import again.",
|
||||||
message_args
|
message_args
|
||||||
)
|
)
|
||||||
: __(
|
: __(
|
||||||
"Successfully updated {0} record out of {1}. Click on Export Errored Rows, fix the errors and import again.",
|
"Successfully updated {0} record out of {1}. Click on Export Errored Rows, fix the errors and import again.",
|
||||||
message_args
|
message_args
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
frm.dashboard.set_headline(message);
|
frm.dashboard.set_headline(message);
|
||||||
@@ -205,7 +236,8 @@ frappe.ui.form.on("Bank Statement Import", {
|
|||||||
},
|
},
|
||||||
|
|
||||||
download_template() {
|
download_template() {
|
||||||
let method = "/api/method/frappe.core.doctype.data_import.data_import.download_template";
|
let method =
|
||||||
|
"/api/method/frappe.core.doctype.data_import.data_import.download_template";
|
||||||
|
|
||||||
open_url_post(method, {
|
open_url_post(method, {
|
||||||
doctype: "Bank Transaction",
|
doctype: "Bank Transaction",
|
||||||
@@ -218,7 +250,7 @@ frappe.ui.form.on("Bank Statement Import", {
|
|||||||
"description",
|
"description",
|
||||||
"reference_number",
|
"reference_number",
|
||||||
"bank_account",
|
"bank_account",
|
||||||
"currency",
|
"currency"
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
@@ -289,7 +321,10 @@ frappe.ui.form.on("Bank Statement Import", {
|
|||||||
show_import_preview(frm, preview_data) {
|
show_import_preview(frm, preview_data) {
|
||||||
let import_log = JSON.parse(frm.doc.statement_import_log || "[]");
|
let import_log = JSON.parse(frm.doc.statement_import_log || "[]");
|
||||||
|
|
||||||
if (frm.import_preview && frm.import_preview.doctype === frm.doc.reference_doctype) {
|
if (
|
||||||
|
frm.import_preview &&
|
||||||
|
frm.import_preview.doctype === frm.doc.reference_doctype
|
||||||
|
) {
|
||||||
frm.import_preview.preview_data = preview_data;
|
frm.import_preview.preview_data = preview_data;
|
||||||
frm.import_preview.import_log = import_log;
|
frm.import_preview.import_log = import_log;
|
||||||
frm.import_preview.refresh();
|
frm.import_preview.refresh();
|
||||||
@@ -305,10 +340,19 @@ frappe.ui.form.on("Bank Statement Import", {
|
|||||||
frm,
|
frm,
|
||||||
events: {
|
events: {
|
||||||
remap_column(changed_map) {
|
remap_column(changed_map) {
|
||||||
let template_options = JSON.parse(frm.doc.template_options || "{}");
|
let template_options = JSON.parse(
|
||||||
template_options.column_to_field_map = template_options.column_to_field_map || {};
|
frm.doc.template_options || "{}"
|
||||||
Object.assign(template_options.column_to_field_map, changed_map);
|
);
|
||||||
frm.set_value("template_options", JSON.stringify(template_options));
|
template_options.column_to_field_map =
|
||||||
|
template_options.column_to_field_map || {};
|
||||||
|
Object.assign(
|
||||||
|
template_options.column_to_field_map,
|
||||||
|
changed_map
|
||||||
|
);
|
||||||
|
frm.set_value(
|
||||||
|
"template_options",
|
||||||
|
JSON.stringify(template_options)
|
||||||
|
);
|
||||||
frm.save().then(() => frm.trigger("import_file"));
|
frm.save().then(() => frm.trigger("import_file"));
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
@@ -342,7 +386,8 @@ frappe.ui.form.on("Bank Statement Import", {
|
|||||||
let other_warnings = [];
|
let other_warnings = [];
|
||||||
for (let warning of warnings) {
|
for (let warning of warnings) {
|
||||||
if (warning.row) {
|
if (warning.row) {
|
||||||
warnings_by_row[warning.row] = warnings_by_row[warning.row] || [];
|
warnings_by_row[warning.row] =
|
||||||
|
warnings_by_row[warning.row] || [];
|
||||||
warnings_by_row[warning.row].push(warning);
|
warnings_by_row[warning.row].push(warning);
|
||||||
} else {
|
} else {
|
||||||
other_warnings.push(warning);
|
other_warnings.push(warning);
|
||||||
@@ -357,7 +402,9 @@ frappe.ui.form.on("Bank Statement Import", {
|
|||||||
if (w.field) {
|
if (w.field) {
|
||||||
let label =
|
let label =
|
||||||
w.field.label +
|
w.field.label +
|
||||||
(w.field.parent !== frm.doc.reference_doctype ? ` (${w.field.parent})` : "");
|
(w.field.parent !== frm.doc.reference_doctype
|
||||||
|
? ` (${w.field.parent})`
|
||||||
|
: "");
|
||||||
return `<li>${label}: ${w.message}</li>`;
|
return `<li>${label}: ${w.message}</li>`;
|
||||||
}
|
}
|
||||||
return `<li>${w.message}</li>`;
|
return `<li>${w.message}</li>`;
|
||||||
@@ -376,9 +423,10 @@ frappe.ui.form.on("Bank Statement Import", {
|
|||||||
.map((warning) => {
|
.map((warning) => {
|
||||||
let header = "";
|
let header = "";
|
||||||
if (warning.col) {
|
if (warning.col) {
|
||||||
let column_number = `<span class="text-uppercase">${__("Column {0}", [
|
let column_number = `<span class="text-uppercase">${__(
|
||||||
warning.col,
|
"Column {0}",
|
||||||
])}</span>`;
|
[warning.col]
|
||||||
|
)}</span>`;
|
||||||
let column_header = columns[warning.col].header_title;
|
let column_header = columns[warning.col].header_title;
|
||||||
header = `${column_number} (${column_header})`;
|
header = `${column_number} (${column_header})`;
|
||||||
}
|
}
|
||||||
@@ -417,28 +465,36 @@ frappe.ui.form.on("Bank Statement Import", {
|
|||||||
let html = "";
|
let html = "";
|
||||||
if (log.success) {
|
if (log.success) {
|
||||||
if (frm.doc.import_type === "Insert New Records") {
|
if (frm.doc.import_type === "Insert New Records") {
|
||||||
html = __("Successfully imported {0}", [
|
html = __(
|
||||||
`<span class="underline">${frappe.utils.get_form_link(
|
"Successfully imported {0}", [
|
||||||
frm.doc.reference_doctype,
|
`<span class="underline">${frappe.utils.get_form_link(
|
||||||
log.docname,
|
frm.doc.reference_doctype,
|
||||||
true
|
log.docname,
|
||||||
)}<span>`,
|
true
|
||||||
]);
|
)}<span>`,
|
||||||
|
]
|
||||||
|
);
|
||||||
} else {
|
} else {
|
||||||
html = __("Successfully updated {0}", [
|
html = __(
|
||||||
`<span class="underline">${frappe.utils.get_form_link(
|
"Successfully updated {0}", [
|
||||||
frm.doc.reference_doctype,
|
`<span class="underline">${frappe.utils.get_form_link(
|
||||||
log.docname,
|
frm.doc.reference_doctype,
|
||||||
true
|
log.docname,
|
||||||
)}<span>`,
|
true
|
||||||
]);
|
)}<span>`,
|
||||||
|
]
|
||||||
|
);
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
let messages = log.messages
|
let messages = log.messages
|
||||||
.map(JSON.parse)
|
.map(JSON.parse)
|
||||||
.map((m) => {
|
.map((m) => {
|
||||||
let title = m.title ? `<strong>${m.title}</strong>` : "";
|
let title = m.title
|
||||||
let message = m.message ? `<div>${m.message}</div>` : "";
|
? `<strong>${m.title}</strong>`
|
||||||
|
: "";
|
||||||
|
let message = m.message
|
||||||
|
? `<div>${m.message}</div>`
|
||||||
|
: "";
|
||||||
return title + message;
|
return title + message;
|
||||||
})
|
})
|
||||||
.join("");
|
.join("");
|
||||||
|
|||||||
@@ -1,34 +1,36 @@
|
|||||||
let imports_in_progress = [];
|
let imports_in_progress = [];
|
||||||
|
|
||||||
frappe.listview_settings["Bank Statement Import"] = {
|
frappe.listview_settings['Bank Statement Import'] = {
|
||||||
onload(listview) {
|
onload(listview) {
|
||||||
frappe.realtime.on("data_import_progress", (data) => {
|
frappe.realtime.on('data_import_progress', data => {
|
||||||
if (!imports_in_progress.includes(data.data_import)) {
|
if (!imports_in_progress.includes(data.data_import)) {
|
||||||
imports_in_progress.push(data.data_import);
|
imports_in_progress.push(data.data_import);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
frappe.realtime.on("data_import_refresh", (data) => {
|
frappe.realtime.on('data_import_refresh', data => {
|
||||||
imports_in_progress = imports_in_progress.filter((d) => d !== data.data_import);
|
imports_in_progress = imports_in_progress.filter(
|
||||||
|
d => d !== data.data_import
|
||||||
|
);
|
||||||
listview.refresh();
|
listview.refresh();
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
get_indicator: function (doc) {
|
get_indicator: function(doc) {
|
||||||
var colors = {
|
var colors = {
|
||||||
Pending: "orange",
|
'Pending': 'orange',
|
||||||
"Not Started": "orange",
|
'Not Started': 'orange',
|
||||||
"Partial Success": "orange",
|
'Partial Success': 'orange',
|
||||||
Success: "green",
|
'Success': 'green',
|
||||||
"In Progress": "orange",
|
'In Progress': 'orange',
|
||||||
Error: "red",
|
'Error': 'red'
|
||||||
};
|
};
|
||||||
let status = doc.status;
|
let status = doc.status;
|
||||||
if (imports_in_progress.includes(doc.name)) {
|
if (imports_in_progress.includes(doc.name)) {
|
||||||
status = "In Progress";
|
status = 'In Progress';
|
||||||
}
|
}
|
||||||
if (status == "Pending") {
|
if (status == 'Pending') {
|
||||||
status = "Not Started";
|
status = 'Not Started';
|
||||||
}
|
}
|
||||||
return [__(status), colors[status], "status,=," + doc.status];
|
return [__(status), colors[status], 'status,=,' + doc.status];
|
||||||
},
|
},
|
||||||
hide_name_column: true,
|
hide_name_column: true
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -112,7 +112,8 @@ class AutoMatchbyPartyNameDescription:
|
|||||||
|
|
||||||
for party in parties:
|
for party in parties:
|
||||||
filters = {"status": "Active"} if party == "Employee" else {"disabled": 0}
|
filters = {"status": "Active"} if party == "Employee" else {"disabled": 0}
|
||||||
names = frappe.get_all(party, filters=filters, pluck=party.lower() + "_name")
|
field = party.lower() + "_name"
|
||||||
|
names = frappe.get_all(party, filters=filters, fields=[f"{field} as party_name", "name"])
|
||||||
|
|
||||||
for field in ["bank_party_name", "description"]:
|
for field in ["bank_party_name", "description"]:
|
||||||
if not self.get(field):
|
if not self.get(field):
|
||||||
@@ -131,7 +132,11 @@ class AutoMatchbyPartyNameDescription:
|
|||||||
|
|
||||||
def fuzzy_search_and_return_result(self, party, names, field) -> Union[Tuple, None]:
|
def fuzzy_search_and_return_result(self, party, names, field) -> Union[Tuple, None]:
|
||||||
skip = False
|
skip = False
|
||||||
result = process.extract(query=self.get(field), choices=names, scorer=fuzz.token_set_ratio)
|
result = process.extract(
|
||||||
|
query=self.get(field),
|
||||||
|
choices={row.get("name"): row.get("party_name") for row in names},
|
||||||
|
scorer=fuzz.token_set_ratio,
|
||||||
|
)
|
||||||
party_name, skip = self.process_fuzzy_result(result)
|
party_name, skip = self.process_fuzzy_result(result)
|
||||||
|
|
||||||
if not party_name:
|
if not party_name:
|
||||||
@@ -149,14 +154,14 @@ class AutoMatchbyPartyNameDescription:
|
|||||||
|
|
||||||
Returns: Result, Skip (whether or not to discontinue matching)
|
Returns: Result, Skip (whether or not to discontinue matching)
|
||||||
"""
|
"""
|
||||||
PARTY, SCORE, CUTOFF = 0, 1, 80
|
SCORE, PARTY_ID, CUTOFF = 1, 2, 80
|
||||||
|
|
||||||
if not result or not len(result):
|
if not result or not len(result):
|
||||||
return None, False
|
return None, False
|
||||||
|
|
||||||
first_result = result[0]
|
first_result = result[0]
|
||||||
if len(result) == 1:
|
if len(result) == 1:
|
||||||
return (first_result[PARTY] if first_result[SCORE] > CUTOFF else None), True
|
return (first_result[PARTY_ID] if first_result[SCORE] > CUTOFF else None), True
|
||||||
|
|
||||||
second_result = result[1]
|
second_result = result[1]
|
||||||
if first_result[SCORE] > CUTOFF:
|
if first_result[SCORE] > CUTOFF:
|
||||||
@@ -165,7 +170,7 @@ class AutoMatchbyPartyNameDescription:
|
|||||||
if first_result[SCORE] == second_result[SCORE]:
|
if first_result[SCORE] == second_result[SCORE]:
|
||||||
return None, True
|
return None, True
|
||||||
|
|
||||||
return first_result[PARTY], True
|
return first_result[PARTY_ID], True
|
||||||
else:
|
else:
|
||||||
return None, False
|
return None, False
|
||||||
|
|
||||||
|
|||||||
@@ -3,7 +3,7 @@
|
|||||||
|
|
||||||
frappe.ui.form.on("Bank Transaction", {
|
frappe.ui.form.on("Bank Transaction", {
|
||||||
onload(frm) {
|
onload(frm) {
|
||||||
frm.set_query("payment_document", "payment_entries", function () {
|
frm.set_query("payment_document", "payment_entries", function() {
|
||||||
const payment_doctypes = frm.events.get_payment_doctypes(frm);
|
const payment_doctypes = frm.events.get_payment_doctypes(frm);
|
||||||
return {
|
return {
|
||||||
filters: {
|
filters: {
|
||||||
@@ -23,7 +23,7 @@ frappe.ui.form.on("Bank Transaction", {
|
|||||||
set_bank_statement_filter(frm);
|
set_bank_statement_filter(frm);
|
||||||
},
|
},
|
||||||
|
|
||||||
setup: function (frm) {
|
setup: function(frm) {
|
||||||
frm.set_query("party_type", function () {
|
frm.set_query("party_type", function () {
|
||||||
return {
|
return {
|
||||||
filters: {
|
filters: {
|
||||||
@@ -33,10 +33,16 @@ frappe.ui.form.on("Bank Transaction", {
|
|||||||
});
|
});
|
||||||
},
|
},
|
||||||
|
|
||||||
get_payment_doctypes: function () {
|
get_payment_doctypes: function() {
|
||||||
// get payment doctypes from all the apps
|
// get payment doctypes from all the apps
|
||||||
return ["Payment Entry", "Journal Entry", "Sales Invoice", "Purchase Invoice", "Bank Transaction"];
|
return [
|
||||||
},
|
"Payment Entry",
|
||||||
|
"Journal Entry",
|
||||||
|
"Sales Invoice",
|
||||||
|
"Purchase Invoice",
|
||||||
|
"Bank Transaction",
|
||||||
|
];
|
||||||
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
frappe.ui.form.on("Bank Transaction Payments", {
|
frappe.ui.form.on("Bank Transaction Payments", {
|
||||||
@@ -48,11 +54,10 @@ frappe.ui.form.on("Bank Transaction Payments", {
|
|||||||
const update_clearance_date = (frm, cdt, cdn) => {
|
const update_clearance_date = (frm, cdt, cdn) => {
|
||||||
if (frm.doc.docstatus === 1) {
|
if (frm.doc.docstatus === 1) {
|
||||||
frappe
|
frappe
|
||||||
.xcall("erpnext.accounts.doctype.bank_transaction.bank_transaction.unclear_reference_payment", {
|
.xcall(
|
||||||
doctype: cdt,
|
"erpnext.accounts.doctype.bank_transaction.bank_transaction.unclear_reference_payment",
|
||||||
docname: cdn,
|
{ doctype: cdt, docname: cdn, bt_name: frm.doc.name }
|
||||||
bt_name: frm.doc.name,
|
)
|
||||||
})
|
|
||||||
.then((e) => {
|
.then((e) => {
|
||||||
if (e == "success") {
|
if (e == "success") {
|
||||||
frappe.show_alert({
|
frappe.show_alert({
|
||||||
|
|||||||
@@ -3,7 +3,6 @@
|
|||||||
|
|
||||||
import frappe
|
import frappe
|
||||||
from frappe import _
|
from frappe import _
|
||||||
from frappe.model.docstatus import DocStatus
|
|
||||||
from frappe.model.document import Document
|
from frappe.model.document import Document
|
||||||
from frappe.utils import flt
|
from frappe.utils import flt
|
||||||
|
|
||||||
@@ -49,24 +48,6 @@ class BankTransaction(Document):
|
|||||||
|
|
||||||
def validate(self):
|
def validate(self):
|
||||||
self.validate_duplicate_references()
|
self.validate_duplicate_references()
|
||||||
self.validate_currency()
|
|
||||||
|
|
||||||
def validate_currency(self):
|
|
||||||
"""
|
|
||||||
Bank Transaction should be on the same currency as the Bank Account.
|
|
||||||
"""
|
|
||||||
if self.currency and self.bank_account:
|
|
||||||
account = frappe.get_cached_value("Bank Account", self.bank_account, "account")
|
|
||||||
account_currency = frappe.get_cached_value("Account", account, "account_currency")
|
|
||||||
|
|
||||||
if self.currency != account_currency:
|
|
||||||
frappe.throw(
|
|
||||||
_(
|
|
||||||
"Transaction currency: {0} cannot be different from Bank Account({1}) currency: {2}"
|
|
||||||
).format(
|
|
||||||
frappe.bold(self.currency), frappe.bold(self.bank_account), frappe.bold(account_currency)
|
|
||||||
)
|
|
||||||
)
|
|
||||||
|
|
||||||
def set_status(self):
|
def set_status(self):
|
||||||
if self.docstatus == 2:
|
if self.docstatus == 2:
|
||||||
@@ -94,13 +75,10 @@ class BankTransaction(Document):
|
|||||||
pe.append(reference)
|
pe.append(reference)
|
||||||
|
|
||||||
def update_allocated_amount(self):
|
def update_allocated_amount(self):
|
||||||
allocated_amount = (
|
self.allocated_amount = (
|
||||||
sum(p.allocated_amount for p in self.payment_entries) if self.payment_entries else 0.0
|
sum(p.allocated_amount for p in self.payment_entries) if self.payment_entries else 0.0
|
||||||
)
|
)
|
||||||
unallocated_amount = abs(flt(self.withdrawal) - flt(self.deposit)) - allocated_amount
|
self.unallocated_amount = abs(flt(self.withdrawal) - flt(self.deposit)) - self.allocated_amount
|
||||||
|
|
||||||
self.allocated_amount = flt(allocated_amount, self.precision("allocated_amount"))
|
|
||||||
self.unallocated_amount = flt(unallocated_amount, self.precision("unallocated_amount"))
|
|
||||||
|
|
||||||
def before_submit(self):
|
def before_submit(self):
|
||||||
self.allocate_payment_entries()
|
self.allocate_payment_entries()
|
||||||
@@ -437,21 +415,3 @@ def unclear_reference_payment(doctype, docname, bt_name):
|
|||||||
bt = frappe.get_doc("Bank Transaction", bt_name)
|
bt = frappe.get_doc("Bank Transaction", bt_name)
|
||||||
set_voucher_clearance(doctype, docname, None, bt)
|
set_voucher_clearance(doctype, docname, None, bt)
|
||||||
return docname
|
return docname
|
||||||
|
|
||||||
|
|
||||||
def remove_from_bank_transaction(doctype, docname):
|
|
||||||
"""Remove a (cancelled) voucher from all Bank Transactions."""
|
|
||||||
for bt_name in get_reconciled_bank_transactions(doctype, docname):
|
|
||||||
bt = frappe.get_doc("Bank Transaction", bt_name)
|
|
||||||
if bt.docstatus == DocStatus.cancelled():
|
|
||||||
continue
|
|
||||||
|
|
||||||
modified = False
|
|
||||||
|
|
||||||
for pe in bt.payment_entries:
|
|
||||||
if pe.payment_document == doctype and pe.payment_entry == docname:
|
|
||||||
bt.remove(pe)
|
|
||||||
modified = True
|
|
||||||
|
|
||||||
if modified:
|
|
||||||
bt.save()
|
|
||||||
|
|||||||
@@ -1,15 +1,15 @@
|
|||||||
// Copyright (c) 2018, Frappe Technologies Pvt. Ltd. and Contributors
|
// Copyright (c) 2018, Frappe Technologies Pvt. Ltd. and Contributors
|
||||||
// License: GNU General Public License v3. See license.txt
|
// License: GNU General Public License v3. See license.txt
|
||||||
|
|
||||||
frappe.listview_settings["Bank Transaction"] = {
|
frappe.listview_settings['Bank Transaction'] = {
|
||||||
add_fields: ["unallocated_amount"],
|
add_fields: ["unallocated_amount"],
|
||||||
get_indicator: function (doc) {
|
get_indicator: function(doc) {
|
||||||
if (doc.docstatus == 2) {
|
if(doc.docstatus == 2) {
|
||||||
return [__("Cancelled"), "red", "docstatus,=,2"];
|
return [__("Cancelled"), "red", "docstatus,=,2"];
|
||||||
} else if (flt(doc.unallocated_amount) <= 0) {
|
} else if(flt(doc.unallocated_amount)<=0) {
|
||||||
return [__("Reconciled"), "green", "unallocated_amount,=,0"];
|
return [__("Reconciled"), "green", "unallocated_amount,=,0"];
|
||||||
} else if (flt(doc.unallocated_amount) > 0) {
|
} else if(flt(doc.unallocated_amount)>0) {
|
||||||
return [__("Unreconciled"), "orange", "unallocated_amount,>,0"];
|
return [__("Unreconciled"), "orange", "unallocated_amount,>,0"];
|
||||||
}
|
}
|
||||||
},
|
}
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -2,10 +2,10 @@
|
|||||||
# See license.txt
|
# See license.txt
|
||||||
|
|
||||||
import json
|
import json
|
||||||
|
import unittest
|
||||||
|
|
||||||
import frappe
|
import frappe
|
||||||
from frappe import utils
|
from frappe import utils
|
||||||
from frappe.model.docstatus import DocStatus
|
|
||||||
from frappe.tests.utils import FrappeTestCase
|
from frappe.tests.utils import FrappeTestCase
|
||||||
|
|
||||||
from erpnext.accounts.doctype.bank_reconciliation_tool.bank_reconciliation_tool import (
|
from erpnext.accounts.doctype.bank_reconciliation_tool.bank_reconciliation_tool import (
|
||||||
@@ -32,16 +32,8 @@ class TestBankTransaction(FrappeTestCase):
|
|||||||
frappe.db.delete(dt)
|
frappe.db.delete(dt)
|
||||||
clear_loan_transactions()
|
clear_loan_transactions()
|
||||||
make_pos_profile()
|
make_pos_profile()
|
||||||
|
add_transactions()
|
||||||
# generate and use a uniq hash identifier for 'Bank Account' and it's linked GL 'Account' to avoid validation error
|
add_vouchers()
|
||||||
uniq_identifier = frappe.generate_hash(length=10)
|
|
||||||
gl_account = create_gl_account("_Test Bank " + uniq_identifier)
|
|
||||||
bank_account = create_bank_account(
|
|
||||||
gl_account=gl_account, bank_account_name="Checking Account " + uniq_identifier
|
|
||||||
)
|
|
||||||
|
|
||||||
add_transactions(bank_account=bank_account)
|
|
||||||
add_vouchers(gl_account=gl_account)
|
|
||||||
|
|
||||||
# This test checks if ERPNext is able to provide a linked payment for a bank transaction based on the amount of the bank transaction.
|
# This test checks if ERPNext is able to provide a linked payment for a bank transaction based on the amount of the bank transaction.
|
||||||
def test_linked_payments(self):
|
def test_linked_payments(self):
|
||||||
@@ -89,29 +81,6 @@ class TestBankTransaction(FrappeTestCase):
|
|||||||
clearance_date = frappe.db.get_value("Payment Entry", payment.name, "clearance_date")
|
clearance_date = frappe.db.get_value("Payment Entry", payment.name, "clearance_date")
|
||||||
self.assertFalse(clearance_date)
|
self.assertFalse(clearance_date)
|
||||||
|
|
||||||
def test_cancel_voucher(self):
|
|
||||||
bank_transaction = frappe.get_doc(
|
|
||||||
"Bank Transaction",
|
|
||||||
dict(description="1512567 BG/000003025 OPSKATTUZWXXX AT776000000098709849 Herr G"),
|
|
||||||
)
|
|
||||||
payment = frappe.get_doc("Payment Entry", dict(party="Mr G", paid_amount=1700))
|
|
||||||
vouchers = json.dumps(
|
|
||||||
[
|
|
||||||
{
|
|
||||||
"payment_doctype": "Payment Entry",
|
|
||||||
"payment_name": payment.name,
|
|
||||||
"amount": bank_transaction.unallocated_amount,
|
|
||||||
}
|
|
||||||
]
|
|
||||||
)
|
|
||||||
reconcile_vouchers(bank_transaction.name, vouchers)
|
|
||||||
payment.reload()
|
|
||||||
payment.cancel()
|
|
||||||
bank_transaction.reload()
|
|
||||||
self.assertEqual(bank_transaction.docstatus, DocStatus.submitted())
|
|
||||||
self.assertEqual(bank_transaction.unallocated_amount, 1700)
|
|
||||||
self.assertEqual(bank_transaction.payment_entries, [])
|
|
||||||
|
|
||||||
# Check if ERPNext can correctly filter a linked payments based on the debit/credit amount
|
# Check if ERPNext can correctly filter a linked payments based on the debit/credit amount
|
||||||
def test_debit_credit_output(self):
|
def test_debit_credit_output(self):
|
||||||
bank_transaction = frappe.get_doc(
|
bank_transaction = frappe.get_doc(
|
||||||
@@ -227,9 +196,7 @@ def clear_loan_transactions():
|
|||||||
frappe.db.delete("Loan Repayment")
|
frappe.db.delete("Loan Repayment")
|
||||||
|
|
||||||
|
|
||||||
def create_bank_account(
|
def create_bank_account(bank_name="Citi Bank", account_name="_Test Bank - _TC"):
|
||||||
bank_name="Citi Bank", gl_account="_Test Bank - _TC", bank_account_name="Checking Account"
|
|
||||||
):
|
|
||||||
try:
|
try:
|
||||||
frappe.get_doc(
|
frappe.get_doc(
|
||||||
{
|
{
|
||||||
@@ -241,35 +208,21 @@ def create_bank_account(
|
|||||||
pass
|
pass
|
||||||
|
|
||||||
try:
|
try:
|
||||||
bank_account = frappe.get_doc(
|
frappe.get_doc(
|
||||||
{
|
{
|
||||||
"doctype": "Bank Account",
|
"doctype": "Bank Account",
|
||||||
"account_name": bank_account_name,
|
"account_name": "Checking Account",
|
||||||
"bank": bank_name,
|
"bank": bank_name,
|
||||||
"account": gl_account,
|
"account": account_name,
|
||||||
}
|
}
|
||||||
).insert(ignore_if_duplicate=True)
|
).insert(ignore_if_duplicate=True)
|
||||||
except frappe.DuplicateEntryError:
|
except frappe.DuplicateEntryError:
|
||||||
pass
|
pass
|
||||||
|
|
||||||
return bank_account.name
|
|
||||||
|
|
||||||
|
def add_transactions():
|
||||||
|
create_bank_account()
|
||||||
|
|
||||||
def create_gl_account(gl_account_name="_Test Bank - _TC"):
|
|
||||||
gl_account = frappe.get_doc(
|
|
||||||
{
|
|
||||||
"doctype": "Account",
|
|
||||||
"company": "_Test Company",
|
|
||||||
"parent_account": "Current Assets - _TC",
|
|
||||||
"account_type": "Bank",
|
|
||||||
"is_group": 0,
|
|
||||||
"account_name": gl_account_name,
|
|
||||||
}
|
|
||||||
).insert()
|
|
||||||
return gl_account.name
|
|
||||||
|
|
||||||
|
|
||||||
def add_transactions(bank_account="_Test Bank - _TC"):
|
|
||||||
doc = frappe.get_doc(
|
doc = frappe.get_doc(
|
||||||
{
|
{
|
||||||
"doctype": "Bank Transaction",
|
"doctype": "Bank Transaction",
|
||||||
@@ -277,7 +230,7 @@ def add_transactions(bank_account="_Test Bank - _TC"):
|
|||||||
"date": "2018-10-23",
|
"date": "2018-10-23",
|
||||||
"deposit": 1200,
|
"deposit": 1200,
|
||||||
"currency": "INR",
|
"currency": "INR",
|
||||||
"bank_account": bank_account,
|
"bank_account": "Checking Account - Citi Bank",
|
||||||
}
|
}
|
||||||
).insert()
|
).insert()
|
||||||
doc.submit()
|
doc.submit()
|
||||||
@@ -289,7 +242,7 @@ def add_transactions(bank_account="_Test Bank - _TC"):
|
|||||||
"date": "2018-10-23",
|
"date": "2018-10-23",
|
||||||
"deposit": 1700,
|
"deposit": 1700,
|
||||||
"currency": "INR",
|
"currency": "INR",
|
||||||
"bank_account": bank_account,
|
"bank_account": "Checking Account - Citi Bank",
|
||||||
}
|
}
|
||||||
).insert()
|
).insert()
|
||||||
doc.submit()
|
doc.submit()
|
||||||
@@ -301,7 +254,7 @@ def add_transactions(bank_account="_Test Bank - _TC"):
|
|||||||
"date": "2018-10-26",
|
"date": "2018-10-26",
|
||||||
"withdrawal": 690,
|
"withdrawal": 690,
|
||||||
"currency": "INR",
|
"currency": "INR",
|
||||||
"bank_account": bank_account,
|
"bank_account": "Checking Account - Citi Bank",
|
||||||
}
|
}
|
||||||
).insert()
|
).insert()
|
||||||
doc.submit()
|
doc.submit()
|
||||||
@@ -313,7 +266,7 @@ def add_transactions(bank_account="_Test Bank - _TC"):
|
|||||||
"date": "2018-10-27",
|
"date": "2018-10-27",
|
||||||
"deposit": 3900,
|
"deposit": 3900,
|
||||||
"currency": "INR",
|
"currency": "INR",
|
||||||
"bank_account": bank_account,
|
"bank_account": "Checking Account - Citi Bank",
|
||||||
}
|
}
|
||||||
).insert()
|
).insert()
|
||||||
doc.submit()
|
doc.submit()
|
||||||
@@ -325,13 +278,13 @@ def add_transactions(bank_account="_Test Bank - _TC"):
|
|||||||
"date": "2018-10-27",
|
"date": "2018-10-27",
|
||||||
"withdrawal": 109080,
|
"withdrawal": 109080,
|
||||||
"currency": "INR",
|
"currency": "INR",
|
||||||
"bank_account": bank_account,
|
"bank_account": "Checking Account - Citi Bank",
|
||||||
}
|
}
|
||||||
).insert()
|
).insert()
|
||||||
doc.submit()
|
doc.submit()
|
||||||
|
|
||||||
|
|
||||||
def add_vouchers(gl_account="_Test Bank - _TC"):
|
def add_vouchers():
|
||||||
try:
|
try:
|
||||||
frappe.get_doc(
|
frappe.get_doc(
|
||||||
{
|
{
|
||||||
@@ -347,7 +300,7 @@ def add_vouchers(gl_account="_Test Bank - _TC"):
|
|||||||
|
|
||||||
pi = make_purchase_invoice(supplier="Conrad Electronic", qty=1, rate=690)
|
pi = make_purchase_invoice(supplier="Conrad Electronic", qty=1, rate=690)
|
||||||
|
|
||||||
pe = get_payment_entry("Purchase Invoice", pi.name, bank_account=gl_account)
|
pe = get_payment_entry("Purchase Invoice", pi.name, bank_account="_Test Bank - _TC")
|
||||||
pe.reference_no = "Conrad Oct 18"
|
pe.reference_no = "Conrad Oct 18"
|
||||||
pe.reference_date = "2018-10-24"
|
pe.reference_date = "2018-10-24"
|
||||||
pe.insert()
|
pe.insert()
|
||||||
@@ -366,14 +319,14 @@ def add_vouchers(gl_account="_Test Bank - _TC"):
|
|||||||
pass
|
pass
|
||||||
|
|
||||||
pi = make_purchase_invoice(supplier="Mr G", qty=1, rate=1200)
|
pi = make_purchase_invoice(supplier="Mr G", qty=1, rate=1200)
|
||||||
pe = get_payment_entry("Purchase Invoice", pi.name, bank_account=gl_account)
|
pe = get_payment_entry("Purchase Invoice", pi.name, bank_account="_Test Bank - _TC")
|
||||||
pe.reference_no = "Herr G Oct 18"
|
pe.reference_no = "Herr G Oct 18"
|
||||||
pe.reference_date = "2018-10-24"
|
pe.reference_date = "2018-10-24"
|
||||||
pe.insert()
|
pe.insert()
|
||||||
pe.submit()
|
pe.submit()
|
||||||
|
|
||||||
pi = make_purchase_invoice(supplier="Mr G", qty=1, rate=1700)
|
pi = make_purchase_invoice(supplier="Mr G", qty=1, rate=1700)
|
||||||
pe = get_payment_entry("Purchase Invoice", pi.name, bank_account=gl_account)
|
pe = get_payment_entry("Purchase Invoice", pi.name, bank_account="_Test Bank - _TC")
|
||||||
pe.reference_no = "Herr G Nov 18"
|
pe.reference_no = "Herr G Nov 18"
|
||||||
pe.reference_date = "2018-11-01"
|
pe.reference_date = "2018-11-01"
|
||||||
pe.insert()
|
pe.insert()
|
||||||
@@ -404,10 +357,10 @@ def add_vouchers(gl_account="_Test Bank - _TC"):
|
|||||||
pass
|
pass
|
||||||
|
|
||||||
pi = make_purchase_invoice(supplier="Poore Simon's", qty=1, rate=3900, is_paid=1, do_not_save=1)
|
pi = make_purchase_invoice(supplier="Poore Simon's", qty=1, rate=3900, is_paid=1, do_not_save=1)
|
||||||
pi.cash_bank_account = gl_account
|
pi.cash_bank_account = "_Test Bank - _TC"
|
||||||
pi.insert()
|
pi.insert()
|
||||||
pi.submit()
|
pi.submit()
|
||||||
pe = get_payment_entry("Purchase Invoice", pi.name, bank_account=gl_account)
|
pe = get_payment_entry("Purchase Invoice", pi.name, bank_account="_Test Bank - _TC")
|
||||||
pe.reference_no = "Poore Simon's Oct 18"
|
pe.reference_no = "Poore Simon's Oct 18"
|
||||||
pe.reference_date = "2018-10-28"
|
pe.reference_date = "2018-10-28"
|
||||||
pe.paid_amount = 690
|
pe.paid_amount = 690
|
||||||
@@ -416,7 +369,7 @@ def add_vouchers(gl_account="_Test Bank - _TC"):
|
|||||||
pe.submit()
|
pe.submit()
|
||||||
|
|
||||||
si = create_sales_invoice(customer="Poore Simon's", qty=1, rate=3900)
|
si = create_sales_invoice(customer="Poore Simon's", qty=1, rate=3900)
|
||||||
pe = get_payment_entry("Sales Invoice", si.name, bank_account=gl_account)
|
pe = get_payment_entry("Sales Invoice", si.name, bank_account="_Test Bank - _TC")
|
||||||
pe.reference_no = "Poore Simon's Oct 18"
|
pe.reference_no = "Poore Simon's Oct 18"
|
||||||
pe.reference_date = "2018-10-28"
|
pe.reference_date = "2018-10-28"
|
||||||
pe.insert()
|
pe.insert()
|
||||||
@@ -439,12 +392,16 @@ def add_vouchers(gl_account="_Test Bank - _TC"):
|
|||||||
if not frappe.db.get_value(
|
if not frappe.db.get_value(
|
||||||
"Mode of Payment Account", {"company": "_Test Company", "parent": "Cash"}
|
"Mode of Payment Account", {"company": "_Test Company", "parent": "Cash"}
|
||||||
):
|
):
|
||||||
mode_of_payment.append("accounts", {"company": "_Test Company", "default_account": gl_account})
|
mode_of_payment.append(
|
||||||
|
"accounts", {"company": "_Test Company", "default_account": "_Test Bank - _TC"}
|
||||||
|
)
|
||||||
mode_of_payment.save()
|
mode_of_payment.save()
|
||||||
|
|
||||||
si = create_sales_invoice(customer="Fayva", qty=1, rate=109080, do_not_save=1)
|
si = create_sales_invoice(customer="Fayva", qty=1, rate=109080, do_not_save=1)
|
||||||
si.is_pos = 1
|
si.is_pos = 1
|
||||||
si.append("payments", {"mode_of_payment": "Cash", "account": gl_account, "amount": 109080})
|
si.append(
|
||||||
|
"payments", {"mode_of_payment": "Cash", "account": "_Test Bank - _TC", "amount": 109080}
|
||||||
|
)
|
||||||
si.insert()
|
si.insert()
|
||||||
si.submit()
|
si.submit()
|
||||||
|
|
||||||
|
|||||||
@@ -0,0 +1,100 @@
|
|||||||
|
// Copyright (c) 2023, Frappe Technologies Pvt. Ltd. and contributors
|
||||||
|
// For license information, please see license.txt
|
||||||
|
|
||||||
|
frappe.ui.form.on("Bisect Accounting Statements", {
|
||||||
|
onload(frm) {
|
||||||
|
frm.trigger("render_heatmap");
|
||||||
|
},
|
||||||
|
refresh(frm) {
|
||||||
|
frm.add_custom_button(__('Bisect Left'), () => {
|
||||||
|
frm.trigger("bisect_left");
|
||||||
|
});
|
||||||
|
|
||||||
|
frm.add_custom_button(__('Bisect Right'), () => {
|
||||||
|
frm.trigger("bisect_right");
|
||||||
|
});
|
||||||
|
|
||||||
|
frm.add_custom_button(__('Up'), () => {
|
||||||
|
frm.trigger("move_up");
|
||||||
|
});
|
||||||
|
frm.add_custom_button(__('Build Tree'), () => {
|
||||||
|
frm.trigger("build_tree");
|
||||||
|
});
|
||||||
|
},
|
||||||
|
render_heatmap(frm) {
|
||||||
|
let bisect_heatmap = frm.get_field("bisect_heatmap").$wrapper;
|
||||||
|
bisect_heatmap.addClass("bisect_heatmap_location");
|
||||||
|
|
||||||
|
// milliseconds in a day
|
||||||
|
let msiad=24*60*60*1000;
|
||||||
|
let datapoints = {};
|
||||||
|
let fr_dt = new Date(frm.doc.from_date).getTime();
|
||||||
|
let to_dt = new Date(frm.doc.to_date).getTime();
|
||||||
|
let bisect_start = new Date(frm.doc.current_from_date).getTime();
|
||||||
|
let bisect_end = new Date(frm.doc.current_to_date).getTime();
|
||||||
|
|
||||||
|
for(let x=fr_dt; x <= to_dt; x+=msiad){
|
||||||
|
let epoch_in_seconds = x/1000;
|
||||||
|
if ((bisect_start <= x) && (x <= bisect_end )) {
|
||||||
|
datapoints[epoch_in_seconds] = 1.0;
|
||||||
|
} else {
|
||||||
|
datapoints[epoch_in_seconds] = 0.0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
new frappe.Chart(".bisect_heatmap_location", {
|
||||||
|
type: "heatmap",
|
||||||
|
data: {
|
||||||
|
dataPoints: datapoints,
|
||||||
|
start: new Date(frm.doc.from_date),
|
||||||
|
end: new Date(frm.doc.to_date),
|
||||||
|
},
|
||||||
|
countLabel: 'Bisecting',
|
||||||
|
discreteDomains: 1,
|
||||||
|
});
|
||||||
|
},
|
||||||
|
bisect_left(frm) {
|
||||||
|
frm.call({
|
||||||
|
doc: frm.doc,
|
||||||
|
method: 'bisect_left',
|
||||||
|
freeze: true,
|
||||||
|
freeze_message: __("Bisecting Left ..."),
|
||||||
|
callback: (r) => {
|
||||||
|
frm.trigger("render_heatmap");
|
||||||
|
}
|
||||||
|
});
|
||||||
|
},
|
||||||
|
bisect_right(frm) {
|
||||||
|
frm.call({
|
||||||
|
doc: frm.doc,
|
||||||
|
freeze: true,
|
||||||
|
freeze_message: __("Bisecting Right ..."),
|
||||||
|
method: 'bisect_right',
|
||||||
|
callback: (r) => {
|
||||||
|
frm.trigger("render_heatmap");
|
||||||
|
}
|
||||||
|
});
|
||||||
|
},
|
||||||
|
move_up(frm) {
|
||||||
|
frm.call({
|
||||||
|
doc: frm.doc,
|
||||||
|
freeze: true,
|
||||||
|
freeze_message: __("Moving up in tree ..."),
|
||||||
|
method: 'move_up',
|
||||||
|
callback: (r) => {
|
||||||
|
frm.trigger("render_heatmap");
|
||||||
|
}
|
||||||
|
});
|
||||||
|
},
|
||||||
|
build_tree(frm) {
|
||||||
|
frm.call({
|
||||||
|
doc: frm.doc,
|
||||||
|
freeze: true,
|
||||||
|
freeze_message: __("Rebuilding BTree for period ..."),
|
||||||
|
method: 'build_tree',
|
||||||
|
callback: (r) => {
|
||||||
|
frm.trigger("render_heatmap");
|
||||||
|
}
|
||||||
|
});
|
||||||
|
},
|
||||||
|
});
|
||||||
@@ -0,0 +1,194 @@
|
|||||||
|
{
|
||||||
|
"actions": [],
|
||||||
|
"allow_rename": 1,
|
||||||
|
"creation": "2023-09-15 21:28:28.054773",
|
||||||
|
"default_view": "List",
|
||||||
|
"doctype": "DocType",
|
||||||
|
"editable_grid": 1,
|
||||||
|
"engine": "InnoDB",
|
||||||
|
"field_order": [
|
||||||
|
"section_break_cvfg",
|
||||||
|
"company",
|
||||||
|
"column_break_hcam",
|
||||||
|
"from_date",
|
||||||
|
"column_break_qxbi",
|
||||||
|
"to_date",
|
||||||
|
"column_break_iwny",
|
||||||
|
"algorithm",
|
||||||
|
"section_break_8ph9",
|
||||||
|
"current_node",
|
||||||
|
"section_break_ngid",
|
||||||
|
"bisect_heatmap",
|
||||||
|
"section_break_hmsy",
|
||||||
|
"bisecting_from",
|
||||||
|
"current_from_date",
|
||||||
|
"column_break_uqyd",
|
||||||
|
"bisecting_to",
|
||||||
|
"current_to_date",
|
||||||
|
"section_break_hbyo",
|
||||||
|
"heading_cppb",
|
||||||
|
"p_l_summary",
|
||||||
|
"column_break_aivo",
|
||||||
|
"balance_sheet_summary",
|
||||||
|
"b_s_summary",
|
||||||
|
"column_break_gvwx",
|
||||||
|
"difference_heading",
|
||||||
|
"difference"
|
||||||
|
],
|
||||||
|
"fields": [
|
||||||
|
{
|
||||||
|
"fieldname": "column_break_qxbi",
|
||||||
|
"fieldtype": "Column Break"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "from_date",
|
||||||
|
"fieldtype": "Datetime",
|
||||||
|
"label": "From Date"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "to_date",
|
||||||
|
"fieldtype": "Datetime",
|
||||||
|
"label": "To Date"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"default": "BFS",
|
||||||
|
"fieldname": "algorithm",
|
||||||
|
"fieldtype": "Select",
|
||||||
|
"label": "Algorithm",
|
||||||
|
"options": "BFS\nDFS"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "column_break_iwny",
|
||||||
|
"fieldtype": "Column Break"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "current_node",
|
||||||
|
"fieldtype": "Link",
|
||||||
|
"label": "Current Node",
|
||||||
|
"options": "Bisect Nodes"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "section_break_hmsy",
|
||||||
|
"fieldtype": "Section Break"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "current_from_date",
|
||||||
|
"fieldtype": "Datetime",
|
||||||
|
"read_only": 1
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "current_to_date",
|
||||||
|
"fieldtype": "Datetime",
|
||||||
|
"read_only": 1
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "column_break_uqyd",
|
||||||
|
"fieldtype": "Column Break"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "section_break_hbyo",
|
||||||
|
"fieldtype": "Section Break"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "p_l_summary",
|
||||||
|
"fieldtype": "Float",
|
||||||
|
"read_only": 1
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "b_s_summary",
|
||||||
|
"fieldtype": "Float",
|
||||||
|
"read_only": 1
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "difference",
|
||||||
|
"fieldtype": "Float",
|
||||||
|
"read_only": 1
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "column_break_aivo",
|
||||||
|
"fieldtype": "Column Break"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "column_break_gvwx",
|
||||||
|
"fieldtype": "Column Break"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "company",
|
||||||
|
"fieldtype": "Link",
|
||||||
|
"label": "Company",
|
||||||
|
"options": "Company"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "column_break_hcam",
|
||||||
|
"fieldtype": "Column Break"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "section_break_ngid",
|
||||||
|
"fieldtype": "Section Break"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "section_break_8ph9",
|
||||||
|
"fieldtype": "Section Break",
|
||||||
|
"hidden": 1
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "bisect_heatmap",
|
||||||
|
"fieldtype": "HTML",
|
||||||
|
"label": "Heatmap"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "heading_cppb",
|
||||||
|
"fieldtype": "Heading",
|
||||||
|
"label": "Profit and Loss Summary"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "balance_sheet_summary",
|
||||||
|
"fieldtype": "Heading",
|
||||||
|
"label": "Balance Sheet Summary"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "difference_heading",
|
||||||
|
"fieldtype": "Heading",
|
||||||
|
"label": "Difference"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "bisecting_from",
|
||||||
|
"fieldtype": "Heading",
|
||||||
|
"label": "Bisecting From"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "bisecting_to",
|
||||||
|
"fieldtype": "Heading",
|
||||||
|
"label": "Bisecting To"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "section_break_cvfg",
|
||||||
|
"fieldtype": "Section Break"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"hide_toolbar": 1,
|
||||||
|
"index_web_pages_for_search": 1,
|
||||||
|
"issingle": 1,
|
||||||
|
"links": [],
|
||||||
|
"modified": "2023-12-01 16:49:54.073890",
|
||||||
|
"modified_by": "Administrator",
|
||||||
|
"module": "Accounts",
|
||||||
|
"name": "Bisect Accounting Statements",
|
||||||
|
"owner": "Administrator",
|
||||||
|
"permissions": [
|
||||||
|
{
|
||||||
|
"create": 1,
|
||||||
|
"delete": 1,
|
||||||
|
"email": 1,
|
||||||
|
"print": 1,
|
||||||
|
"read": 1,
|
||||||
|
"role": "Administrator",
|
||||||
|
"share": 1,
|
||||||
|
"write": 1
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"read_only": 1,
|
||||||
|
"sort_field": "modified",
|
||||||
|
"sort_order": "DESC",
|
||||||
|
"states": []
|
||||||
|
}
|
||||||
@@ -0,0 +1,226 @@
|
|||||||
|
# Copyright (c) 2023, Frappe Technologies Pvt. Ltd. and contributors
|
||||||
|
# For license information, please see license.txt
|
||||||
|
|
||||||
|
import datetime
|
||||||
|
from collections import deque
|
||||||
|
from math import floor
|
||||||
|
|
||||||
|
import frappe
|
||||||
|
from dateutil.relativedelta import relativedelta
|
||||||
|
from frappe import _
|
||||||
|
from frappe.model.document import Document
|
||||||
|
from frappe.utils import getdate
|
||||||
|
from frappe.utils.data import guess_date_format
|
||||||
|
|
||||||
|
|
||||||
|
class BisectAccountingStatements(Document):
|
||||||
|
# begin: auto-generated types
|
||||||
|
# This code is auto-generated. Do not modify anything in this block.
|
||||||
|
|
||||||
|
from typing import TYPE_CHECKING
|
||||||
|
|
||||||
|
if TYPE_CHECKING:
|
||||||
|
from frappe.types import DF
|
||||||
|
|
||||||
|
algorithm: DF.Literal["BFS", "DFS"]
|
||||||
|
b_s_summary: DF.Float
|
||||||
|
company: DF.Link | None
|
||||||
|
current_from_date: DF.Datetime | None
|
||||||
|
current_node: DF.Link | None
|
||||||
|
current_to_date: DF.Datetime | None
|
||||||
|
difference: DF.Float
|
||||||
|
from_date: DF.Datetime | None
|
||||||
|
p_l_summary: DF.Float
|
||||||
|
to_date: DF.Datetime | None
|
||||||
|
# end: auto-generated types
|
||||||
|
|
||||||
|
def validate(self):
|
||||||
|
self.validate_dates()
|
||||||
|
|
||||||
|
def validate_dates(self):
|
||||||
|
if getdate(self.from_date) > getdate(self.to_date):
|
||||||
|
frappe.throw(
|
||||||
|
_("From Date: {0} cannot be greater than To date: {1}").format(
|
||||||
|
frappe.bold(self.from_date), frappe.bold(self.to_date)
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
|
def bfs(self, from_date: datetime, to_date: datetime):
|
||||||
|
# Make Root node
|
||||||
|
node = frappe.new_doc("Bisect Nodes")
|
||||||
|
node.root = None
|
||||||
|
node.period_from_date = from_date
|
||||||
|
node.period_to_date = to_date
|
||||||
|
node.insert()
|
||||||
|
|
||||||
|
period_queue = deque([node])
|
||||||
|
while period_queue:
|
||||||
|
cur_node = period_queue.popleft()
|
||||||
|
delta = cur_node.period_to_date - cur_node.period_from_date
|
||||||
|
if delta.days == 0:
|
||||||
|
continue
|
||||||
|
else:
|
||||||
|
cur_floor = floor(delta.days / 2)
|
||||||
|
next_to_date = cur_node.period_from_date + relativedelta(days=+cur_floor)
|
||||||
|
left_node = frappe.new_doc("Bisect Nodes")
|
||||||
|
left_node.period_from_date = cur_node.period_from_date
|
||||||
|
left_node.period_to_date = next_to_date
|
||||||
|
left_node.root = cur_node.name
|
||||||
|
left_node.generated = False
|
||||||
|
left_node.insert()
|
||||||
|
cur_node.left_child = left_node.name
|
||||||
|
period_queue.append(left_node)
|
||||||
|
|
||||||
|
next_from_date = cur_node.period_from_date + relativedelta(days=+(cur_floor + 1))
|
||||||
|
right_node = frappe.new_doc("Bisect Nodes")
|
||||||
|
right_node.period_from_date = next_from_date
|
||||||
|
right_node.period_to_date = cur_node.period_to_date
|
||||||
|
right_node.root = cur_node.name
|
||||||
|
right_node.generated = False
|
||||||
|
right_node.insert()
|
||||||
|
cur_node.right_child = right_node.name
|
||||||
|
period_queue.append(right_node)
|
||||||
|
|
||||||
|
cur_node.save()
|
||||||
|
|
||||||
|
def dfs(self, from_date: datetime, to_date: datetime):
|
||||||
|
# Make Root node
|
||||||
|
node = frappe.new_doc("Bisect Nodes")
|
||||||
|
node.root = None
|
||||||
|
node.period_from_date = from_date
|
||||||
|
node.period_to_date = to_date
|
||||||
|
node.insert()
|
||||||
|
|
||||||
|
period_stack = [node]
|
||||||
|
while period_stack:
|
||||||
|
cur_node = period_stack.pop()
|
||||||
|
delta = cur_node.period_to_date - cur_node.period_from_date
|
||||||
|
if delta.days == 0:
|
||||||
|
continue
|
||||||
|
else:
|
||||||
|
cur_floor = floor(delta.days / 2)
|
||||||
|
next_to_date = cur_node.period_from_date + relativedelta(days=+cur_floor)
|
||||||
|
left_node = frappe.new_doc("Bisect Nodes")
|
||||||
|
left_node.period_from_date = cur_node.period_from_date
|
||||||
|
left_node.period_to_date = next_to_date
|
||||||
|
left_node.root = cur_node.name
|
||||||
|
left_node.generated = False
|
||||||
|
left_node.insert()
|
||||||
|
cur_node.left_child = left_node.name
|
||||||
|
period_stack.append(left_node)
|
||||||
|
|
||||||
|
next_from_date = cur_node.period_from_date + relativedelta(days=+(cur_floor + 1))
|
||||||
|
right_node = frappe.new_doc("Bisect Nodes")
|
||||||
|
right_node.period_from_date = next_from_date
|
||||||
|
right_node.period_to_date = cur_node.period_to_date
|
||||||
|
right_node.root = cur_node.name
|
||||||
|
right_node.generated = False
|
||||||
|
right_node.insert()
|
||||||
|
cur_node.right_child = right_node.name
|
||||||
|
period_stack.append(right_node)
|
||||||
|
|
||||||
|
cur_node.save()
|
||||||
|
|
||||||
|
@frappe.whitelist()
|
||||||
|
def build_tree(self):
|
||||||
|
frappe.db.delete("Bisect Nodes")
|
||||||
|
|
||||||
|
# Convert str to datetime format
|
||||||
|
dt_format = guess_date_format(self.from_date)
|
||||||
|
from_date = datetime.datetime.strptime(self.from_date, dt_format)
|
||||||
|
to_date = datetime.datetime.strptime(self.to_date, dt_format)
|
||||||
|
|
||||||
|
if self.algorithm == "BFS":
|
||||||
|
self.bfs(from_date, to_date)
|
||||||
|
|
||||||
|
if self.algorithm == "DFS":
|
||||||
|
self.dfs(from_date, to_date)
|
||||||
|
|
||||||
|
# set root as current node
|
||||||
|
root = frappe.db.get_all("Bisect Nodes", filters={"root": ["is", "not set"]})[0]
|
||||||
|
self.get_report_summary()
|
||||||
|
self.current_node = root.name
|
||||||
|
self.current_from_date = self.from_date
|
||||||
|
self.current_to_date = self.to_date
|
||||||
|
self.save()
|
||||||
|
|
||||||
|
def get_report_summary(self):
|
||||||
|
filters = {
|
||||||
|
"company": self.company,
|
||||||
|
"filter_based_on": "Date Range",
|
||||||
|
"period_start_date": self.current_from_date,
|
||||||
|
"period_end_date": self.current_to_date,
|
||||||
|
"periodicity": "Yearly",
|
||||||
|
}
|
||||||
|
pl_summary = frappe.get_doc("Report", "Profit and Loss Statement")
|
||||||
|
self.p_l_summary = pl_summary.execute_script_report(filters=filters)[5]
|
||||||
|
bs_summary = frappe.get_doc("Report", "Balance Sheet")
|
||||||
|
self.b_s_summary = bs_summary.execute_script_report(filters=filters)[5]
|
||||||
|
self.difference = abs(self.p_l_summary - self.b_s_summary)
|
||||||
|
|
||||||
|
def update_node(self):
|
||||||
|
current_node = frappe.get_doc("Bisect Nodes", self.current_node)
|
||||||
|
current_node.balance_sheet_summary = self.b_s_summary
|
||||||
|
current_node.profit_loss_summary = self.p_l_summary
|
||||||
|
current_node.difference = self.difference
|
||||||
|
current_node.generated = True
|
||||||
|
current_node.save()
|
||||||
|
|
||||||
|
def current_node_has_summary_info(self):
|
||||||
|
"Assertion method"
|
||||||
|
return frappe.db.get_value("Bisect Nodes", self.current_node, "generated")
|
||||||
|
|
||||||
|
def fetch_summary_info_from_current_node(self):
|
||||||
|
current_node = frappe.get_doc("Bisect Nodes", self.current_node)
|
||||||
|
self.p_l_summary = current_node.balance_sheet_summary
|
||||||
|
self.b_s_summary = current_node.profit_loss_summary
|
||||||
|
self.difference = abs(self.p_l_summary - self.b_s_summary)
|
||||||
|
|
||||||
|
def fetch_or_calculate(self):
|
||||||
|
if self.current_node_has_summary_info():
|
||||||
|
self.fetch_summary_info_from_current_node()
|
||||||
|
else:
|
||||||
|
self.get_report_summary()
|
||||||
|
self.update_node()
|
||||||
|
|
||||||
|
@frappe.whitelist()
|
||||||
|
def bisect_left(self):
|
||||||
|
if self.current_node is not None:
|
||||||
|
cur_node = frappe.get_doc("Bisect Nodes", self.current_node)
|
||||||
|
if cur_node.left_child is not None:
|
||||||
|
lft_node = frappe.get_doc("Bisect Nodes", cur_node.left_child)
|
||||||
|
self.current_node = cur_node.left_child
|
||||||
|
self.current_from_date = lft_node.period_from_date
|
||||||
|
self.current_to_date = lft_node.period_to_date
|
||||||
|
self.fetch_or_calculate()
|
||||||
|
self.save()
|
||||||
|
else:
|
||||||
|
frappe.msgprint(_("No more children on Left"))
|
||||||
|
|
||||||
|
@frappe.whitelist()
|
||||||
|
def bisect_right(self):
|
||||||
|
if self.current_node is not None:
|
||||||
|
cur_node = frappe.get_doc("Bisect Nodes", self.current_node)
|
||||||
|
if cur_node.right_child is not None:
|
||||||
|
rgt_node = frappe.get_doc("Bisect Nodes", cur_node.right_child)
|
||||||
|
self.current_node = cur_node.right_child
|
||||||
|
self.current_from_date = rgt_node.period_from_date
|
||||||
|
self.current_to_date = rgt_node.period_to_date
|
||||||
|
self.fetch_or_calculate()
|
||||||
|
self.save()
|
||||||
|
else:
|
||||||
|
frappe.msgprint(_("No more children on Right"))
|
||||||
|
|
||||||
|
@frappe.whitelist()
|
||||||
|
def move_up(self):
|
||||||
|
if self.current_node is not None:
|
||||||
|
cur_node = frappe.get_doc("Bisect Nodes", self.current_node)
|
||||||
|
if cur_node.root is not None:
|
||||||
|
root = frappe.get_doc("Bisect Nodes", cur_node.root)
|
||||||
|
self.current_node = cur_node.root
|
||||||
|
self.current_from_date = root.period_from_date
|
||||||
|
self.current_to_date = root.period_to_date
|
||||||
|
self.fetch_or_calculate()
|
||||||
|
self.save()
|
||||||
|
else:
|
||||||
|
frappe.msgprint(_("Reached Root"))
|
||||||
@@ -0,0 +1,9 @@
|
|||||||
|
# Copyright (c) 2023, Frappe Technologies Pvt. Ltd. and Contributors
|
||||||
|
# See license.txt
|
||||||
|
|
||||||
|
# import frappe
|
||||||
|
from frappe.tests.utils import FrappeTestCase
|
||||||
|
|
||||||
|
|
||||||
|
class TestBisectAccountingStatements(FrappeTestCase):
|
||||||
|
pass
|
||||||
8
erpnext/accounts/doctype/bisect_nodes/bisect_nodes.js
Normal file
8
erpnext/accounts/doctype/bisect_nodes/bisect_nodes.js
Normal file
@@ -0,0 +1,8 @@
|
|||||||
|
// Copyright (c) 2023, Frappe Technologies Pvt. Ltd. and contributors
|
||||||
|
// For license information, please see license.txt
|
||||||
|
|
||||||
|
// frappe.ui.form.on("Bisect Nodes", {
|
||||||
|
// refresh(frm) {
|
||||||
|
|
||||||
|
// },
|
||||||
|
// });
|
||||||
97
erpnext/accounts/doctype/bisect_nodes/bisect_nodes.json
Normal file
97
erpnext/accounts/doctype/bisect_nodes/bisect_nodes.json
Normal file
@@ -0,0 +1,97 @@
|
|||||||
|
{
|
||||||
|
"actions": [],
|
||||||
|
"autoname": "autoincrement",
|
||||||
|
"creation": "2023-09-27 14:56:38.112462",
|
||||||
|
"default_view": "List",
|
||||||
|
"doctype": "DocType",
|
||||||
|
"editable_grid": 1,
|
||||||
|
"engine": "InnoDB",
|
||||||
|
"field_order": [
|
||||||
|
"root",
|
||||||
|
"left_child",
|
||||||
|
"right_child",
|
||||||
|
"period_from_date",
|
||||||
|
"period_to_date",
|
||||||
|
"difference",
|
||||||
|
"balance_sheet_summary",
|
||||||
|
"profit_loss_summary",
|
||||||
|
"generated"
|
||||||
|
],
|
||||||
|
"fields": [
|
||||||
|
{
|
||||||
|
"fieldname": "root",
|
||||||
|
"fieldtype": "Link",
|
||||||
|
"label": "Root",
|
||||||
|
"options": "Bisect Nodes"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "left_child",
|
||||||
|
"fieldtype": "Link",
|
||||||
|
"label": "Left Child",
|
||||||
|
"options": "Bisect Nodes"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "right_child",
|
||||||
|
"fieldtype": "Link",
|
||||||
|
"label": "Right Child",
|
||||||
|
"options": "Bisect Nodes"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "period_from_date",
|
||||||
|
"fieldtype": "Datetime",
|
||||||
|
"label": "Period_from_date"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "period_to_date",
|
||||||
|
"fieldtype": "Datetime",
|
||||||
|
"label": "Period To Date"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "difference",
|
||||||
|
"fieldtype": "Float",
|
||||||
|
"label": "Difference"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "balance_sheet_summary",
|
||||||
|
"fieldtype": "Float",
|
||||||
|
"label": "Balance Sheet Summary"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "profit_loss_summary",
|
||||||
|
"fieldtype": "Float",
|
||||||
|
"label": "Profit and Loss Summary"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"default": "0",
|
||||||
|
"fieldname": "generated",
|
||||||
|
"fieldtype": "Check",
|
||||||
|
"label": "Generated"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"index_web_pages_for_search": 1,
|
||||||
|
"links": [],
|
||||||
|
"modified": "2023-12-01 17:46:12.437996",
|
||||||
|
"modified_by": "Administrator",
|
||||||
|
"module": "Accounts",
|
||||||
|
"name": "Bisect Nodes",
|
||||||
|
"naming_rule": "Autoincrement",
|
||||||
|
"owner": "Administrator",
|
||||||
|
"permissions": [
|
||||||
|
{
|
||||||
|
"create": 1,
|
||||||
|
"delete": 1,
|
||||||
|
"email": 1,
|
||||||
|
"export": 1,
|
||||||
|
"print": 1,
|
||||||
|
"read": 1,
|
||||||
|
"report": 1,
|
||||||
|
"role": "Administrator",
|
||||||
|
"share": 1,
|
||||||
|
"write": 1
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"read_only": 1,
|
||||||
|
"sort_field": "modified",
|
||||||
|
"sort_order": "DESC",
|
||||||
|
"states": []
|
||||||
|
}
|
||||||
29
erpnext/accounts/doctype/bisect_nodes/bisect_nodes.py
Normal file
29
erpnext/accounts/doctype/bisect_nodes/bisect_nodes.py
Normal file
@@ -0,0 +1,29 @@
|
|||||||
|
# Copyright (c) 2023, Frappe Technologies Pvt. Ltd. and contributors
|
||||||
|
# For license information, please see license.txt
|
||||||
|
|
||||||
|
# import frappe
|
||||||
|
from frappe.model.document import Document
|
||||||
|
|
||||||
|
|
||||||
|
class BisectNodes(Document):
|
||||||
|
# begin: auto-generated types
|
||||||
|
# This code is auto-generated. Do not modify anything in this block.
|
||||||
|
|
||||||
|
from typing import TYPE_CHECKING
|
||||||
|
|
||||||
|
if TYPE_CHECKING:
|
||||||
|
from frappe.types import DF
|
||||||
|
|
||||||
|
balance_sheet_summary: DF.Float
|
||||||
|
difference: DF.Float
|
||||||
|
generated: DF.Check
|
||||||
|
left_child: DF.Link | None
|
||||||
|
name: DF.Int | None
|
||||||
|
period_from_date: DF.Datetime | None
|
||||||
|
period_to_date: DF.Datetime | None
|
||||||
|
profit_loss_summary: DF.Float
|
||||||
|
right_child: DF.Link | None
|
||||||
|
root: DF.Link | None
|
||||||
|
# end: auto-generated types
|
||||||
|
|
||||||
|
pass
|
||||||
@@ -5,5 +5,5 @@
|
|||||||
from frappe.tests.utils import FrappeTestCase
|
from frappe.tests.utils import FrappeTestCase
|
||||||
|
|
||||||
|
|
||||||
class TestPlantFloor(FrappeTestCase):
|
class TestBisectNodes(FrappeTestCase):
|
||||||
pass
|
pass
|
||||||
@@ -2,48 +2,48 @@
|
|||||||
// For license information, please see license.txt
|
// For license information, please see license.txt
|
||||||
frappe.provide("erpnext.accounts.dimensions");
|
frappe.provide("erpnext.accounts.dimensions");
|
||||||
|
|
||||||
frappe.ui.form.on("Budget", {
|
frappe.ui.form.on('Budget', {
|
||||||
onload: function (frm) {
|
onload: function(frm) {
|
||||||
frm.set_query("account", "accounts", function () {
|
frm.set_query("account", "accounts", function() {
|
||||||
return {
|
return {
|
||||||
filters: {
|
filters: {
|
||||||
company: frm.doc.company,
|
company: frm.doc.company,
|
||||||
report_type: "Profit and Loss",
|
report_type: "Profit and Loss",
|
||||||
is_group: 0,
|
is_group: 0
|
||||||
},
|
}
|
||||||
};
|
};
|
||||||
});
|
});
|
||||||
|
|
||||||
frm.set_query("monthly_distribution", function () {
|
frm.set_query("monthly_distribution", function() {
|
||||||
return {
|
return {
|
||||||
filters: {
|
filters: {
|
||||||
fiscal_year: frm.doc.fiscal_year,
|
fiscal_year: frm.doc.fiscal_year
|
||||||
},
|
}
|
||||||
};
|
};
|
||||||
});
|
});
|
||||||
|
|
||||||
erpnext.accounts.dimensions.setup_dimension_filters(frm, frm.doctype);
|
erpnext.accounts.dimensions.setup_dimension_filters(frm, frm.doctype);
|
||||||
},
|
},
|
||||||
|
|
||||||
refresh: function (frm) {
|
refresh: function(frm) {
|
||||||
frm.trigger("toggle_reqd_fields");
|
frm.trigger("toggle_reqd_fields")
|
||||||
},
|
},
|
||||||
|
|
||||||
budget_against: function (frm) {
|
budget_against: function(frm) {
|
||||||
frm.trigger("set_null_value");
|
frm.trigger("set_null_value")
|
||||||
frm.trigger("toggle_reqd_fields");
|
frm.trigger("toggle_reqd_fields")
|
||||||
},
|
},
|
||||||
|
|
||||||
set_null_value: function (frm) {
|
set_null_value: function(frm) {
|
||||||
if (frm.doc.budget_against == "Cost Center") {
|
if(frm.doc.budget_against == 'Cost Center') {
|
||||||
frm.set_value("project", null);
|
frm.set_value('project', null)
|
||||||
} else {
|
} else {
|
||||||
frm.set_value("cost_center", null);
|
frm.set_value('cost_center', null)
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
toggle_reqd_fields: function (frm) {
|
toggle_reqd_fields: function(frm) {
|
||||||
frm.toggle_reqd("cost_center", frm.doc.budget_against == "Cost Center");
|
frm.toggle_reqd("cost_center", frm.doc.budget_against=="Cost Center");
|
||||||
frm.toggle_reqd("project", frm.doc.budget_against == "Project");
|
frm.toggle_reqd("project", frm.doc.budget_against=="Project");
|
||||||
},
|
}
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -1,10 +1,11 @@
|
|||||||
// Copyright (c) 2015, Frappe Technologies Pvt. Ltd. and Contributors
|
// Copyright (c) 2015, Frappe Technologies Pvt. Ltd. and Contributors
|
||||||
// License: GNU General Public License v3. See license.txt
|
// License: GNU General Public License v3. See license.txt
|
||||||
|
|
||||||
frappe.ui.form.on("Cashier Closing", {
|
frappe.ui.form.on('Cashier Closing', {
|
||||||
setup: function (frm) {
|
|
||||||
|
setup: function(frm){
|
||||||
if (frm.doc.user == "" || frm.doc.user == null) {
|
if (frm.doc.user == "" || frm.doc.user == null) {
|
||||||
frm.doc.user = frappe.session.user;
|
frm.doc.user = frappe.session.user;
|
||||||
}
|
}
|
||||||
},
|
}
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
frappe.ui.form.on("Chart of Accounts Importer", {
|
frappe.ui.form.on('Chart of Accounts Importer', {
|
||||||
onload: function (frm) {
|
onload: function (frm) {
|
||||||
frm.set_value("company", "");
|
frm.set_value("company", "");
|
||||||
frm.set_value("import_file", "");
|
frm.set_value("import_file", "");
|
||||||
@@ -8,34 +8,31 @@ frappe.ui.form.on("Chart of Accounts Importer", {
|
|||||||
frm.disable_save();
|
frm.disable_save();
|
||||||
|
|
||||||
// make company mandatory
|
// make company mandatory
|
||||||
frm.set_df_property("company", "reqd", frm.doc.company ? 0 : 1);
|
frm.set_df_property('company', 'reqd', frm.doc.company ? 0 : 1);
|
||||||
frm.set_df_property("import_file_section", "hidden", frm.doc.company ? 0 : 1);
|
frm.set_df_property('import_file_section', 'hidden', frm.doc.company ? 0 : 1);
|
||||||
|
|
||||||
if (frm.doc.import_file) {
|
if (frm.doc.import_file) {
|
||||||
frappe.run_serially([
|
frappe.run_serially([
|
||||||
() => generate_tree_preview(frm),
|
() => generate_tree_preview(frm),
|
||||||
() => create_import_button(frm),
|
() => create_import_button(frm),
|
||||||
() => frm.set_df_property("chart_preview", "hidden", 0),
|
() => frm.set_df_property('chart_preview', 'hidden', 0)
|
||||||
]);
|
]);
|
||||||
}
|
}
|
||||||
|
|
||||||
frm.set_df_property(
|
frm.set_df_property('chart_preview', 'hidden',
|
||||||
"chart_preview",
|
$(frm.fields_dict['chart_tree'].wrapper).html()!="" ? 0 : 1);
|
||||||
"hidden",
|
|
||||||
$(frm.fields_dict["chart_tree"].wrapper).html() != "" ? 0 : 1
|
|
||||||
);
|
|
||||||
},
|
},
|
||||||
|
|
||||||
download_template: function (frm) {
|
download_template: function(frm) {
|
||||||
var d = new frappe.ui.Dialog({
|
var d = new frappe.ui.Dialog({
|
||||||
title: __("Download Template"),
|
title: __("Download Template"),
|
||||||
fields: [
|
fields: [
|
||||||
{
|
{
|
||||||
label: "File Type",
|
label : "File Type",
|
||||||
fieldname: "file_type",
|
fieldname: "file_type",
|
||||||
fieldtype: "Select",
|
fieldtype: "Select",
|
||||||
reqd: 1,
|
reqd: 1,
|
||||||
options: ["Excel", "CSV"],
|
options: ["Excel", "CSV"]
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
label: "Template Type",
|
label: "Template Type",
|
||||||
@@ -44,27 +41,21 @@ frappe.ui.form.on("Chart of Accounts Importer", {
|
|||||||
reqd: 1,
|
reqd: 1,
|
||||||
options: ["Sample Template", "Blank Template"],
|
options: ["Sample Template", "Blank Template"],
|
||||||
change: () => {
|
change: () => {
|
||||||
let template_type = d.get_value("template_type");
|
let template_type = d.get_value('template_type');
|
||||||
|
|
||||||
if (template_type === "Sample Template") {
|
if (template_type === "Sample Template") {
|
||||||
d.set_df_property(
|
d.set_df_property('template_type', 'description',
|
||||||
"template_type",
|
|
||||||
"description",
|
|
||||||
`The Sample Template contains all the required accounts pre filled in the template.
|
`The Sample Template contains all the required accounts pre filled in the template.
|
||||||
You can add more accounts or change existing accounts in the template as per your choice.`
|
You can add more accounts or change existing accounts in the template as per your choice.`);
|
||||||
);
|
|
||||||
} else {
|
} else {
|
||||||
d.set_df_property(
|
d.set_df_property('template_type', 'description',
|
||||||
"template_type",
|
|
||||||
"description",
|
|
||||||
`The Blank Template contains just the account type and root type required to build the Chart
|
`The Blank Template contains just the account type and root type required to build the Chart
|
||||||
of Accounts. Please enter the account names and add more rows as per your requirement.`
|
of Accounts. Please enter the account names and add more rows as per your requirement.`);
|
||||||
);
|
|
||||||
}
|
}
|
||||||
},
|
}
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
label: "Company",
|
label : "Company",
|
||||||
fieldname: "company",
|
fieldname: "company",
|
||||||
fieldtype: "Link",
|
fieldtype: "Link",
|
||||||
reqd: 1,
|
reqd: 1,
|
||||||
@@ -72,25 +63,25 @@ frappe.ui.form.on("Chart of Accounts Importer", {
|
|||||||
default: frm.doc.company,
|
default: frm.doc.company,
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
primary_action: function () {
|
primary_action: function() {
|
||||||
let data = d.get_values();
|
let data = d.get_values();
|
||||||
|
|
||||||
if (!data.template_type) {
|
if (!data.template_type) {
|
||||||
frappe.throw(__("Please select <b>Template Type</b> to download template"));
|
frappe.throw(__('Please select <b>Template Type</b> to download template'));
|
||||||
}
|
}
|
||||||
|
|
||||||
open_url_post(
|
open_url_post(
|
||||||
"/api/method/erpnext.accounts.doctype.chart_of_accounts_importer.chart_of_accounts_importer.download_template",
|
'/api/method/erpnext.accounts.doctype.chart_of_accounts_importer.chart_of_accounts_importer.download_template',
|
||||||
{
|
{
|
||||||
file_type: data.file_type,
|
file_type: data.file_type,
|
||||||
template_type: data.template_type,
|
template_type: data.template_type,
|
||||||
company: data.company,
|
company: data.company
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
|
|
||||||
d.hide();
|
d.hide();
|
||||||
},
|
},
|
||||||
primary_action_label: __("Download"),
|
primary_action_label: __('Download')
|
||||||
});
|
});
|
||||||
d.show();
|
d.show();
|
||||||
},
|
},
|
||||||
@@ -98,7 +89,7 @@ frappe.ui.form.on("Chart of Accounts Importer", {
|
|||||||
import_file: function (frm) {
|
import_file: function (frm) {
|
||||||
if (!frm.doc.import_file) {
|
if (!frm.doc.import_file) {
|
||||||
frm.page.set_indicator("");
|
frm.page.set_indicator("");
|
||||||
$(frm.fields_dict["chart_tree"].wrapper).empty(); // empty wrapper on removing file
|
$(frm.fields_dict['chart_tree'].wrapper).empty(); // empty wrapper on removing file
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
@@ -108,97 +99,89 @@ frappe.ui.form.on("Chart of Accounts Importer", {
|
|||||||
frappe.call({
|
frappe.call({
|
||||||
method: "erpnext.accounts.doctype.chart_of_accounts_importer.chart_of_accounts_importer.validate_company",
|
method: "erpnext.accounts.doctype.chart_of_accounts_importer.chart_of_accounts_importer.validate_company",
|
||||||
args: {
|
args: {
|
||||||
company: frm.doc.company,
|
company: frm.doc.company
|
||||||
},
|
},
|
||||||
callback: function (r) {
|
callback: function(r) {
|
||||||
if (r.message === false) {
|
if(r.message===false) {
|
||||||
frm.set_value("company", "");
|
frm.set_value("company", "");
|
||||||
frappe.throw(
|
frappe.throw(__("Transactions against the Company already exist! Chart of Accounts can only be imported for a Company with no transactions."));
|
||||||
__(
|
|
||||||
"Transactions against the Company already exist! Chart of Accounts can only be imported for a Company with no transactions."
|
|
||||||
)
|
|
||||||
);
|
|
||||||
} else {
|
} else {
|
||||||
frm.trigger("refresh");
|
frm.trigger("refresh");
|
||||||
}
|
}
|
||||||
},
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
},
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
var create_import_button = function (frm) {
|
var create_import_button = function(frm) {
|
||||||
frm.page
|
frm.page.set_primary_action(__("Import"), function () {
|
||||||
.set_primary_action(__("Import"), function () {
|
|
||||||
return frappe.call({
|
|
||||||
method: "erpnext.accounts.doctype.chart_of_accounts_importer.chart_of_accounts_importer.import_coa",
|
|
||||||
args: {
|
|
||||||
file_name: frm.doc.import_file,
|
|
||||||
company: frm.doc.company,
|
|
||||||
},
|
|
||||||
freeze: true,
|
|
||||||
freeze_message: __("Creating Accounts..."),
|
|
||||||
callback: function (r) {
|
|
||||||
if (!r.exc) {
|
|
||||||
clearInterval(frm.page["interval"]);
|
|
||||||
frm.page.set_indicator(__("Import Successful"), "blue");
|
|
||||||
create_reset_button(frm);
|
|
||||||
}
|
|
||||||
},
|
|
||||||
});
|
|
||||||
})
|
|
||||||
.addClass("btn btn-primary");
|
|
||||||
};
|
|
||||||
|
|
||||||
var create_reset_button = function (frm) {
|
|
||||||
frm.page
|
|
||||||
.set_primary_action(__("Reset"), function () {
|
|
||||||
frm.page.clear_primary_action();
|
|
||||||
delete frm.page["show_import_button"];
|
|
||||||
frm.reload_doc();
|
|
||||||
})
|
|
||||||
.addClass("btn btn-primary");
|
|
||||||
};
|
|
||||||
|
|
||||||
var validate_coa = function (frm) {
|
|
||||||
if (frm.doc.import_file) {
|
|
||||||
let parent = __("All Accounts");
|
|
||||||
return frappe.call({
|
return frappe.call({
|
||||||
method: "erpnext.accounts.doctype.chart_of_accounts_importer.chart_of_accounts_importer.get_coa",
|
method: "erpnext.accounts.doctype.chart_of_accounts_importer.chart_of_accounts_importer.import_coa",
|
||||||
args: {
|
args: {
|
||||||
file_name: frm.doc.import_file,
|
file_name: frm.doc.import_file,
|
||||||
parent: parent,
|
company: frm.doc.company
|
||||||
doctype: "Chart of Accounts Importer",
|
|
||||||
file_type: frm.doc.file_type,
|
|
||||||
for_validate: 1,
|
|
||||||
},
|
},
|
||||||
callback: function (r) {
|
freeze: true,
|
||||||
if (r.message["show_import_button"]) {
|
freeze_message: __("Creating Accounts..."),
|
||||||
frm.page["show_import_button"] = Boolean(r.message["show_import_button"]);
|
callback: function(r) {
|
||||||
|
if (!r.exc) {
|
||||||
|
clearInterval(frm.page["interval"]);
|
||||||
|
frm.page.set_indicator(__('Import Successful'), 'blue');
|
||||||
|
create_reset_button(frm);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}).addClass('btn btn-primary');
|
||||||
|
};
|
||||||
|
|
||||||
|
var create_reset_button = function(frm) {
|
||||||
|
frm.page.set_primary_action(__("Reset"), function () {
|
||||||
|
frm.page.clear_primary_action();
|
||||||
|
delete frm.page["show_import_button"];
|
||||||
|
frm.reload_doc();
|
||||||
|
}).addClass('btn btn-primary');
|
||||||
|
};
|
||||||
|
|
||||||
|
var validate_coa = function(frm) {
|
||||||
|
if (frm.doc.import_file) {
|
||||||
|
let parent = __('All Accounts');
|
||||||
|
return frappe.call({
|
||||||
|
'method': 'erpnext.accounts.doctype.chart_of_accounts_importer.chart_of_accounts_importer.get_coa',
|
||||||
|
'args': {
|
||||||
|
file_name: frm.doc.import_file,
|
||||||
|
parent: parent,
|
||||||
|
doctype: 'Chart of Accounts Importer',
|
||||||
|
file_type: frm.doc.file_type,
|
||||||
|
for_validate: 1
|
||||||
},
|
},
|
||||||
|
callback: function(r) {
|
||||||
|
if (r.message['show_import_button']) {
|
||||||
|
frm.page['show_import_button'] = Boolean(r.message['show_import_button']);
|
||||||
|
}
|
||||||
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
var generate_tree_preview = function (frm) {
|
var generate_tree_preview = function(frm) {
|
||||||
let parent = __("All Accounts");
|
let parent = __('All Accounts');
|
||||||
$(frm.fields_dict["chart_tree"].wrapper).empty(); // empty wrapper to load new data
|
$(frm.fields_dict['chart_tree'].wrapper).empty(); // empty wrapper to load new data
|
||||||
|
|
||||||
// generate tree structure based on the csv data
|
// generate tree structure based on the csv data
|
||||||
return new frappe.ui.Tree({
|
return new frappe.ui.Tree({
|
||||||
parent: $(frm.fields_dict["chart_tree"].wrapper),
|
parent: $(frm.fields_dict['chart_tree'].wrapper),
|
||||||
label: parent,
|
label: parent,
|
||||||
expandable: true,
|
expandable: true,
|
||||||
method: "erpnext.accounts.doctype.chart_of_accounts_importer.chart_of_accounts_importer.get_coa",
|
method: 'erpnext.accounts.doctype.chart_of_accounts_importer.chart_of_accounts_importer.get_coa',
|
||||||
args: {
|
args: {
|
||||||
file_name: frm.doc.import_file,
|
file_name: frm.doc.import_file,
|
||||||
parent: parent,
|
parent: parent,
|
||||||
doctype: "Chart of Accounts Importer",
|
doctype: 'Chart of Accounts Importer',
|
||||||
file_type: frm.doc.file_type,
|
file_type: frm.doc.file_type
|
||||||
},
|
},
|
||||||
onclick: function (node) {
|
onclick: function(node) {
|
||||||
parent = node.value;
|
parent = node.value;
|
||||||
},
|
}
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -3,20 +3,18 @@
|
|||||||
|
|
||||||
frappe.provide("erpnext.cheque_print");
|
frappe.provide("erpnext.cheque_print");
|
||||||
|
|
||||||
frappe.ui.form.on("Cheque Print Template", {
|
frappe.ui.form.on('Cheque Print Template', {
|
||||||
refresh: function (frm) {
|
refresh: function(frm) {
|
||||||
if (!frm.doc.__islocal) {
|
if(!frm.doc.__islocal) {
|
||||||
frm.add_custom_button(
|
frm.add_custom_button(frm.doc.has_print_format?__("Update Print Format"):__("Create Print Format"),
|
||||||
frm.doc.has_print_format ? __("Update Print Format") : __("Create Print Format"),
|
function() {
|
||||||
function () {
|
|
||||||
erpnext.cheque_print.view_cheque_print(frm);
|
erpnext.cheque_print.view_cheque_print(frm);
|
||||||
}
|
}).addClass("btn-primary");
|
||||||
).addClass("btn-primary");
|
|
||||||
|
|
||||||
$(frm.fields_dict.cheque_print_preview.wrapper).empty();
|
$(frm.fields_dict.cheque_print_preview.wrapper).empty()
|
||||||
|
|
||||||
var template =
|
|
||||||
'<div style="position: relative; overflow-x: scroll;">\
|
var template = '<div style="position: relative; overflow-x: scroll;">\
|
||||||
<div id="cheque_preview" style="width: {{ cheque_width }}cm; \
|
<div id="cheque_preview" style="width: {{ cheque_width }}cm; \
|
||||||
height: {{ cheque_height }}cm;\
|
height: {{ cheque_height }}cm;\
|
||||||
background-repeat: no-repeat;\
|
background-repeat: no-repeat;\
|
||||||
@@ -50,30 +48,30 @@ frappe.ui.form.on("Cheque Print Template", {
|
|||||||
</div>\
|
</div>\
|
||||||
</div>';
|
</div>';
|
||||||
|
|
||||||
$(frappe.render(template, frm.doc)).appendTo(frm.fields_dict.cheque_print_preview.wrapper);
|
$(frappe.render(template, frm.doc)).appendTo(frm.fields_dict.cheque_print_preview.wrapper)
|
||||||
|
|
||||||
if (frm.doc.scanned_cheque) {
|
if (frm.doc.scanned_cheque) {
|
||||||
$(frm.fields_dict.cheque_print_preview.wrapper)
|
$(frm.fields_dict.cheque_print_preview.wrapper).find("#cheque_preview").css('background-image', 'url(' + frm.doc.scanned_cheque + ')');
|
||||||
.find("#cheque_preview")
|
|
||||||
.css("background-image", "url(" + frm.doc.scanned_cheque + ")");
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
erpnext.cheque_print.view_cheque_print = function (frm) {
|
|
||||||
|
erpnext.cheque_print.view_cheque_print = function(frm) {
|
||||||
frappe.call({
|
frappe.call({
|
||||||
method: "erpnext.accounts.doctype.cheque_print_template.cheque_print_template.create_or_update_cheque_print_format",
|
method: "erpnext.accounts.doctype.cheque_print_template.cheque_print_template.create_or_update_cheque_print_format",
|
||||||
args: {
|
args:{
|
||||||
template_name: frm.doc.name,
|
"template_name": frm.doc.name
|
||||||
},
|
},
|
||||||
callback: function (r) {
|
callback: function(r) {
|
||||||
if (!r.exe && !frm.doc.has_print_format) {
|
if (!r.exe && !frm.doc.has_print_format) {
|
||||||
var doc = frappe.model.sync(r.message);
|
var doc = frappe.model.sync(r.message);
|
||||||
frappe.set_route("Form", r.message.doctype, r.message.name);
|
frappe.set_route("Form", r.message.doctype, r.message.name);
|
||||||
} else {
|
|
||||||
frappe.msgprint(__("Print settings updated in respective print format"));
|
|
||||||
}
|
}
|
||||||
},
|
else {
|
||||||
});
|
frappe.msgprint(__("Print settings updated in respective print format"))
|
||||||
};
|
}
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|||||||
@@ -3,80 +3,75 @@
|
|||||||
|
|
||||||
frappe.provide("erpnext.accounts");
|
frappe.provide("erpnext.accounts");
|
||||||
|
|
||||||
frappe.ui.form.on("Cost Center", {
|
|
||||||
onload: function (frm) {
|
|
||||||
frm.set_query("parent_cost_center", function () {
|
frappe.ui.form.on('Cost Center', {
|
||||||
|
onload: function(frm) {
|
||||||
|
frm.set_query("parent_cost_center", function() {
|
||||||
return {
|
return {
|
||||||
filters: {
|
filters: {
|
||||||
company: frm.doc.company,
|
company: frm.doc.company,
|
||||||
is_group: 1,
|
is_group: 1
|
||||||
},
|
}
|
||||||
};
|
}
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
refresh: function (frm) {
|
refresh: function(frm) {
|
||||||
if (!frm.is_new()) {
|
if (!frm.is_new()) {
|
||||||
frm.add_custom_button(__("Update Cost Center Name / Number"), function () {
|
frm.add_custom_button(__('Update Cost Center Name / Number'), function () {
|
||||||
frm.trigger("update_cost_center_number");
|
frm.trigger("update_cost_center_number");
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
let intro_txt = "";
|
let intro_txt = '';
|
||||||
let doc = frm.doc;
|
let doc = frm.doc;
|
||||||
frm.toggle_display("cost_center_name", doc.__islocal);
|
frm.toggle_display('cost_center_name', doc.__islocal);
|
||||||
frm.toggle_enable(["is_group", "company"], doc.__islocal);
|
frm.toggle_enable(['is_group', 'company'], doc.__islocal);
|
||||||
|
|
||||||
if (!doc.__islocal && doc.is_group == 1) {
|
if(!doc.__islocal && doc.is_group==1) {
|
||||||
intro_txt += __(
|
intro_txt += __('Note: This Cost Center is a Group. Cannot make accounting entries against groups.');
|
||||||
"Note: This Cost Center is a Group. Cannot make accounting entries against groups."
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
frm.events.hide_unhide_group_ledger(frm);
|
frm.events.hide_unhide_group_ledger(frm);
|
||||||
|
|
||||||
frm.toggle_display("sb1", doc.is_group == 0);
|
frm.toggle_display('sb1', doc.is_group==0);
|
||||||
frm.set_intro(intro_txt);
|
frm.set_intro(intro_txt);
|
||||||
|
|
||||||
if (!frm.doc.__islocal) {
|
if(!frm.doc.__islocal) {
|
||||||
frm.add_custom_button(__("Chart of Cost Centers"), function () {
|
frm.add_custom_button(__('Chart of Cost Centers'),
|
||||||
frappe.set_route("Tree", "Cost Center");
|
function() { frappe.set_route("Tree", "Cost Center"); });
|
||||||
});
|
|
||||||
|
|
||||||
frm.add_custom_button(__("Budget"), function () {
|
frm.add_custom_button(__('Budget'),
|
||||||
frappe.set_route("List", "Budget", { cost_center: frm.doc.name });
|
function() { frappe.set_route("List", "Budget", {'cost_center': frm.doc.name}); });
|
||||||
});
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
update_cost_center_number: function (frm) {
|
update_cost_center_number: function(frm) {
|
||||||
var d = new frappe.ui.Dialog({
|
var d = new frappe.ui.Dialog({
|
||||||
title: __("Update Cost Center Name / Number"),
|
title: __('Update Cost Center Name / Number'),
|
||||||
fields: [
|
fields: [
|
||||||
{
|
{
|
||||||
label: "Cost Center Name",
|
"label": "Cost Center Name",
|
||||||
fieldname: "cost_center_name",
|
"fieldname": "cost_center_name",
|
||||||
fieldtype: "Data",
|
"fieldtype": "Data",
|
||||||
reqd: 1,
|
"reqd": 1,
|
||||||
default: frm.doc.cost_center_name,
|
"default": frm.doc.cost_center_name
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
label: "Cost Center Number",
|
"label": "Cost Center Number",
|
||||||
fieldname: "cost_center_number",
|
"fieldname": "cost_center_number",
|
||||||
fieldtype: "Data",
|
"fieldtype": "Data",
|
||||||
default: frm.doc.cost_center_number,
|
"default": frm.doc.cost_center_number
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
label: __("Merge with existing"),
|
"label": __("Merge with existing"),
|
||||||
fieldname: "merge",
|
"fieldname": "merge",
|
||||||
fieldtype: "Check",
|
"fieldtype": "Check",
|
||||||
default: 0,
|
"default": 0
|
||||||
},
|
}
|
||||||
],
|
],
|
||||||
primary_action: function () {
|
primary_action: function() {
|
||||||
let data = d.get_values();
|
let data = d.get_values();
|
||||||
if (
|
if(data.cost_center_name === frm.doc.cost_center_name && data.cost_center_number === frm.doc.cost_center_number) {
|
||||||
data.cost_center_name === frm.doc.cost_center_name &&
|
|
||||||
data.cost_center_number === frm.doc.cost_center_number
|
|
||||||
) {
|
|
||||||
d.hide();
|
d.hide();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@@ -88,12 +83,12 @@ frappe.ui.form.on("Cost Center", {
|
|||||||
cost_center_name: data.cost_center_name,
|
cost_center_name: data.cost_center_name,
|
||||||
cost_center_number: cstr(data.cost_center_number),
|
cost_center_number: cstr(data.cost_center_number),
|
||||||
company: frm.doc.company,
|
company: frm.doc.company,
|
||||||
merge: data.merge,
|
merge: data.merge
|
||||||
},
|
},
|
||||||
callback: function (r) {
|
callback: function(r) {
|
||||||
frappe.dom.unfreeze();
|
frappe.dom.unfreeze();
|
||||||
if (!r.exc) {
|
if(!r.exc) {
|
||||||
if (r.message) {
|
if(r.message) {
|
||||||
frappe.set_route("Form", "Cost Center", r.message);
|
frappe.set_route("Form", "Cost Center", r.message);
|
||||||
} else {
|
} else {
|
||||||
frm.set_value("cost_center_name", data.cost_center_name);
|
frm.set_value("cost_center_name", data.cost_center_name);
|
||||||
@@ -101,42 +96,44 @@ frappe.ui.form.on("Cost Center", {
|
|||||||
}
|
}
|
||||||
d.hide();
|
d.hide();
|
||||||
}
|
}
|
||||||
},
|
}
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
primary_action_label: __("Update"),
|
primary_action_label: __('Update')
|
||||||
});
|
});
|
||||||
d.show();
|
d.show();
|
||||||
},
|
},
|
||||||
|
|
||||||
parent_cost_center(frm) {
|
parent_cost_center(frm) {
|
||||||
if (!frm.doc.company) {
|
if(!frm.doc.company) {
|
||||||
frappe.msgprint(__("Please enter company name first"));
|
frappe.msgprint(__('Please enter company name first'));
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
hide_unhide_group_ledger(frm) {
|
hide_unhide_group_ledger(frm) {
|
||||||
let doc = frm.doc;
|
let doc = frm.doc;
|
||||||
if (doc.is_group == 1) {
|
if (doc.is_group == 1) {
|
||||||
frm.add_custom_button(__("Convert to Non-Group"), () => frm.events.convert_to_ledger(frm));
|
frm.add_custom_button(__('Convert to Non-Group'),
|
||||||
|
() => frm.events.convert_to_ledger(frm));
|
||||||
} else if (doc.is_group == 0) {
|
} else if (doc.is_group == 0) {
|
||||||
frm.add_custom_button(__("Convert to Group"), () => frm.events.convert_to_group(frm));
|
frm.add_custom_button(__('Convert to Group'),
|
||||||
|
() => frm.events.convert_to_group(frm));
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
convert_to_group(frm) {
|
convert_to_group(frm) {
|
||||||
frm.call("convert_ledger_to_group").then((r) => {
|
frm.call('convert_ledger_to_group').then(r => {
|
||||||
if (r.message === 1) {
|
if(r.message === 1) {
|
||||||
frm.refresh();
|
frm.refresh();
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
|
|
||||||
convert_to_ledger(frm) {
|
convert_to_ledger(frm) {
|
||||||
frm.call("convert_group_to_ledger").then((r) => {
|
frm.call('convert_group_to_ledger').then(r => {
|
||||||
if (r.message === 1) {
|
if(r.message === 1) {
|
||||||
frm.refresh();
|
frm.refresh();
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
},
|
}
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -41,7 +41,7 @@
|
|||||||
"in_list_view": 1,
|
"in_list_view": 1,
|
||||||
"in_standard_filter": 1,
|
"in_standard_filter": 1,
|
||||||
"label": "Cost Center Number",
|
"label": "Cost Center Number",
|
||||||
"read_only_depends_on": "eval:!doc.__islocal"
|
"read_only": 1
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"fieldname": "parent_cost_center",
|
"fieldname": "parent_cost_center",
|
||||||
@@ -170,4 +170,4 @@
|
|||||||
"sort_field": "modified",
|
"sort_field": "modified",
|
||||||
"sort_order": "ASC",
|
"sort_order": "ASC",
|
||||||
"states": []
|
"states": []
|
||||||
}
|
}
|
||||||
@@ -1,84 +1,54 @@
|
|||||||
frappe.treeview_settings["Cost Center"] = {
|
frappe.treeview_settings["Cost Center"] = {
|
||||||
breadcrumb: "Accounts",
|
breadcrumb: "Accounts",
|
||||||
get_tree_root: false,
|
get_tree_root: false,
|
||||||
filters: [
|
filters: [{
|
||||||
{
|
fieldname: "company",
|
||||||
fieldname: "company",
|
fieldtype:"Select",
|
||||||
fieldtype: "Select",
|
options: erpnext.utils.get_tree_options("company"),
|
||||||
options: erpnext.utils.get_tree_options("company"),
|
label: __("Company"),
|
||||||
label: __("Company"),
|
default: erpnext.utils.get_tree_default("company")
|
||||||
default: erpnext.utils.get_tree_default("company"),
|
}],
|
||||||
},
|
|
||||||
],
|
|
||||||
root_label: "Cost Centers",
|
root_label: "Cost Centers",
|
||||||
get_tree_nodes: "erpnext.accounts.utils.get_children",
|
get_tree_nodes: 'erpnext.accounts.utils.get_children',
|
||||||
add_tree_node: "erpnext.accounts.utils.add_cc",
|
add_tree_node: 'erpnext.accounts.utils.add_cc',
|
||||||
menu_items: [
|
menu_items:[
|
||||||
{
|
{
|
||||||
label: __("New Company"),
|
label: __('New Company'),
|
||||||
action: function () {
|
action: function() { frappe.new_doc("Company", true) },
|
||||||
frappe.new_doc("Company", true);
|
condition: 'frappe.boot.user.can_create.indexOf("Company") !== -1'
|
||||||
},
|
}
|
||||||
condition: 'frappe.boot.user.can_create.indexOf("Company") !== -1',
|
|
||||||
},
|
|
||||||
],
|
],
|
||||||
fields: [
|
fields:[
|
||||||
{ fieldtype: "Data", fieldname: "cost_center_name", label: __("New Cost Center Name"), reqd: true },
|
{fieldtype:'Data', fieldname:'cost_center_name', label:__('New Cost Center Name'), reqd:true},
|
||||||
{
|
{fieldtype:'Check', fieldname:'is_group', label:__('Is Group'),
|
||||||
fieldtype: "Check",
|
description:__('Further cost centers can be made under Groups but entries can be made against non-Groups')},
|
||||||
fieldname: "is_group",
|
{fieldtype:'Data', fieldname:'cost_center_number', label:__('Cost Center Number'),
|
||||||
label: __("Is Group"),
|
description: __("Number of new Cost Center, it will be included in the cost center name as a prefix")}
|
||||||
description: __(
|
|
||||||
"Further cost centers can be made under Groups but entries can be made against non-Groups"
|
|
||||||
),
|
|
||||||
},
|
|
||||||
{
|
|
||||||
fieldtype: "Data",
|
|
||||||
fieldname: "cost_center_number",
|
|
||||||
label: __("Cost Center Number"),
|
|
||||||
description: __(
|
|
||||||
"Number of new Cost Center, it will be included in the cost center name as a prefix"
|
|
||||||
),
|
|
||||||
},
|
|
||||||
],
|
],
|
||||||
ignore_fields: ["parent_cost_center"],
|
ignore_fields:["parent_cost_center"],
|
||||||
onload: function (treeview) {
|
onload: function(treeview) {
|
||||||
function get_company() {
|
function get_company() {
|
||||||
return treeview.page.fields_dict.company.get_value();
|
return treeview.page.fields_dict.company.get_value();
|
||||||
}
|
}
|
||||||
|
|
||||||
// tools
|
// tools
|
||||||
treeview.page.add_inner_button(
|
treeview.page.add_inner_button(__("Chart of Accounts"), function() {
|
||||||
__("Chart of Accounts"),
|
frappe.set_route('Tree', 'Account', {company: get_company()});
|
||||||
function () {
|
}, __('View'));
|
||||||
frappe.set_route("Tree", "Account", { company: get_company() });
|
|
||||||
},
|
|
||||||
__("View")
|
|
||||||
);
|
|
||||||
|
|
||||||
// make
|
// make
|
||||||
treeview.page.add_inner_button(
|
treeview.page.add_inner_button(__("Budget List"), function() {
|
||||||
__("Budget List"),
|
frappe.set_route('List', 'Budget', {company: get_company()});
|
||||||
function () {
|
}, __('Budget'));
|
||||||
frappe.set_route("List", "Budget", { company: get_company() });
|
|
||||||
},
|
|
||||||
__("Budget")
|
|
||||||
);
|
|
||||||
|
|
||||||
treeview.page.add_inner_button(
|
treeview.page.add_inner_button(__("Monthly Distribution"), function() {
|
||||||
__("Monthly Distribution"),
|
frappe.set_route('List', 'Monthly Distribution', {company: get_company()});
|
||||||
function () {
|
}, __('Budget'));
|
||||||
frappe.set_route("List", "Monthly Distribution", { company: get_company() });
|
|
||||||
},
|
|
||||||
__("Budget")
|
|
||||||
);
|
|
||||||
|
|
||||||
treeview.page.add_inner_button(
|
treeview.page.add_inner_button(__("Budget Variance Report"), function() {
|
||||||
__("Budget Variance Report"),
|
frappe.set_route('query-report', 'Budget Variance Report', {company: get_company()});
|
||||||
function () {
|
}, __('Budget'));
|
||||||
frappe.set_route("query-report", "Budget Variance Report", { company: get_company() });
|
|
||||||
},
|
}
|
||||||
__("Budget")
|
|
||||||
);
|
}
|
||||||
},
|
|
||||||
};
|
|
||||||
|
|||||||
@@ -1,24 +1,19 @@
|
|||||||
// Copyright (c) 2022, Frappe Technologies Pvt. Ltd. and contributors
|
// Copyright (c) 2022, Frappe Technologies Pvt. Ltd. and contributors
|
||||||
// For license information, please see license.txt
|
// For license information, please see license.txt
|
||||||
|
|
||||||
frappe.ui.form.on("Cost Center Allocation", {
|
frappe.ui.form.on('Cost Center Allocation', {
|
||||||
setup: function (frm) {
|
setup: function(frm) {
|
||||||
frm.set_query("main_cost_center", function () {
|
let filters = {"is_group": 0};
|
||||||
return {
|
if (frm.doc.company) {
|
||||||
filters: {
|
$.extend(filters, {
|
||||||
company: frm.doc.company,
|
"company": frm.doc.company
|
||||||
is_group: 0,
|
});
|
||||||
},
|
}
|
||||||
};
|
|
||||||
});
|
|
||||||
|
|
||||||
frm.set_query("cost_center", "allocation_percentages", function () {
|
frm.set_query('main_cost_center', function() {
|
||||||
return {
|
return {
|
||||||
filters: {
|
filters: filters
|
||||||
company: frm.doc.company,
|
|
||||||
is_group: 0,
|
|
||||||
},
|
|
||||||
};
|
};
|
||||||
});
|
});
|
||||||
},
|
}
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -1,41 +1,44 @@
|
|||||||
// Copyright (c) 2018, Frappe Technologies Pvt. Ltd. and contributors
|
// Copyright (c) 2018, Frappe Technologies Pvt. Ltd. and contributors
|
||||||
// For license information, please see license.txt
|
// For license information, please see license.txt
|
||||||
|
|
||||||
frappe.ui.form.on("Coupon Code", {
|
frappe.ui.form.on('Coupon Code', {
|
||||||
setup: function (frm) {
|
setup: function(frm) {
|
||||||
frm.set_query("pricing_rule", function () {
|
frm.set_query("pricing_rule", function() {
|
||||||
return {
|
return {
|
||||||
filters: [["Pricing Rule", "coupon_code_based", "=", "1"]],
|
filters: [
|
||||||
|
["Pricing Rule","coupon_code_based", "=", "1"]
|
||||||
|
]
|
||||||
};
|
};
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
coupon_name: function (frm) {
|
coupon_name:function(frm){
|
||||||
if (frm.doc.__islocal === 1) {
|
if (frm.doc.__islocal===1) {
|
||||||
frm.trigger("make_coupon_code");
|
frm.trigger("make_coupon_code");
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
coupon_type: function (frm) {
|
coupon_type:function(frm){
|
||||||
if (frm.doc.__islocal === 1) {
|
if (frm.doc.__islocal===1) {
|
||||||
frm.trigger("make_coupon_code");
|
frm.trigger("make_coupon_code");
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
make_coupon_code: function (frm) {
|
make_coupon_code: function(frm) {
|
||||||
var coupon_name = frm.doc.coupon_name;
|
var coupon_name=frm.doc.coupon_name;
|
||||||
var coupon_code;
|
var coupon_code;
|
||||||
if (frm.doc.coupon_type == "Gift Card") {
|
if (frm.doc.coupon_type=='Gift Card') {
|
||||||
coupon_code = Math.random().toString(12).substring(2, 12).toUpperCase();
|
coupon_code=Math.random().toString(12).substring(2, 12).toUpperCase();
|
||||||
} else if (frm.doc.coupon_type == "Promotional") {
|
|
||||||
coupon_name = coupon_name.replace(/\s/g, "");
|
|
||||||
coupon_code = coupon_name.toUpperCase().slice(0, 8);
|
|
||||||
}
|
}
|
||||||
frm.doc.coupon_code = coupon_code;
|
else if(frm.doc.coupon_type=='Promotional'){
|
||||||
frm.refresh_field("coupon_code");
|
coupon_name=coupon_name.replace(/\s/g,'');
|
||||||
|
coupon_code=coupon_name.toUpperCase().slice(0,8);
|
||||||
|
}
|
||||||
|
frm.doc.coupon_code=coupon_code;
|
||||||
|
frm.refresh_field('coupon_code');
|
||||||
},
|
},
|
||||||
refresh: function (frm) {
|
refresh: function(frm) {
|
||||||
if (frm.doc.pricing_rule) {
|
if (frm.doc.pricing_rule) {
|
||||||
frm.add_custom_button(__("Add/Edit Coupon Conditions"), function () {
|
frm.add_custom_button(__("Add/Edit Coupon Conditions"), function(){
|
||||||
frappe.set_route("Form", "Pricing Rule", frm.doc.pricing_rule);
|
frappe.set_route("Form", "Pricing Rule", frm.doc.pricing_rule);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
},
|
}
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -1,27 +1,28 @@
|
|||||||
// Copyright (c) 2022, Frappe Technologies Pvt. Ltd. and contributors
|
// Copyright (c) 2022, Frappe Technologies Pvt. Ltd. and contributors
|
||||||
// For license information, please see license.txt
|
// For license information, please see license.txt
|
||||||
|
|
||||||
frappe.ui.form.on("Currency Exchange Settings", {
|
frappe.ui.form.on('Currency Exchange Settings', {
|
||||||
service_provider: function (frm) {
|
service_provider: function(frm) {
|
||||||
if (frm.doc.service_provider == "exchangerate.host") {
|
if (frm.doc.service_provider == "exchangerate.host") {
|
||||||
let result = ["result"];
|
let result = ['result'];
|
||||||
let params = {
|
let params = {
|
||||||
date: "{transaction_date}",
|
date: '{transaction_date}',
|
||||||
from: "{from_currency}",
|
from: '{from_currency}',
|
||||||
to: "{to_currency}",
|
to: '{to_currency}'
|
||||||
};
|
};
|
||||||
add_param(frm, "https://api.exchangerate.host/convert", params, result);
|
add_param(frm, "https://api.exchangerate.host/convert", params, result);
|
||||||
} else if (frm.doc.service_provider == "frankfurter.app") {
|
} else if (frm.doc.service_provider == "frankfurter.app") {
|
||||||
let result = ["rates", "{to_currency}"];
|
let result = ['rates', '{to_currency}'];
|
||||||
let params = {
|
let params = {
|
||||||
base: "{from_currency}",
|
base: '{from_currency}',
|
||||||
symbols: "{to_currency}",
|
symbols: '{to_currency}'
|
||||||
};
|
};
|
||||||
add_param(frm, "https://frankfurter.app/{transaction_date}", params, result);
|
add_param(frm, "https://frankfurter.app/{transaction_date}", params, result);
|
||||||
}
|
}
|
||||||
},
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
||||||
function add_param(frm, api, params, result) {
|
function add_param(frm, api, params, result) {
|
||||||
var row;
|
var row;
|
||||||
frm.clear_table("req_params");
|
frm.clear_table("req_params");
|
||||||
@@ -29,13 +30,13 @@ function add_param(frm, api, params, result) {
|
|||||||
|
|
||||||
frm.doc.api_endpoint = api;
|
frm.doc.api_endpoint = api;
|
||||||
|
|
||||||
$.each(params, function (key, value) {
|
$.each(params, function(key, value) {
|
||||||
row = frm.add_child("req_params");
|
row = frm.add_child("req_params");
|
||||||
row.key = key;
|
row.key = key;
|
||||||
row.value = value;
|
row.value = value;
|
||||||
});
|
});
|
||||||
|
|
||||||
$.each(result, function (key, value) {
|
$.each(result, function(key, value) {
|
||||||
row = frm.add_child("result_key");
|
row = frm.add_child("result_key");
|
||||||
row.key = value;
|
row.key = value;
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -10,7 +10,7 @@ frappe.ui.form.on("Dunning", {
|
|||||||
company: frm.doc.company,
|
company: frm.doc.company,
|
||||||
customer: frm.doc.customer,
|
customer: frm.doc.customer,
|
||||||
outstanding_amount: [">", 0],
|
outstanding_amount: [">", 0],
|
||||||
status: "Overdue",
|
status: "Overdue"
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
});
|
});
|
||||||
@@ -19,16 +19,16 @@ frappe.ui.form.on("Dunning", {
|
|||||||
filters: {
|
filters: {
|
||||||
company: frm.doc.company,
|
company: frm.doc.company,
|
||||||
root_type: "Income",
|
root_type: "Income",
|
||||||
is_group: 0,
|
is_group: 0
|
||||||
},
|
}
|
||||||
};
|
};
|
||||||
});
|
});
|
||||||
frm.set_query("cost_center", () => {
|
frm.set_query("cost_center", () => {
|
||||||
return {
|
return {
|
||||||
filters: {
|
filters: {
|
||||||
company: frm.doc.company,
|
company: frm.doc.company,
|
||||||
is_group: 0,
|
is_group: 0
|
||||||
},
|
}
|
||||||
};
|
};
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -51,8 +51,7 @@ frappe.ui.form.on("Dunning", {
|
|||||||
__("Payment"),
|
__("Payment"),
|
||||||
function () {
|
function () {
|
||||||
frm.events.make_payment_entry(frm);
|
frm.events.make_payment_entry(frm);
|
||||||
},
|
}, __("Create")
|
||||||
__("Create")
|
|
||||||
);
|
);
|
||||||
frm.page.set_inner_btn_group_as_primary(__("Create"));
|
frm.page.set_inner_btn_group_as_primary(__("Create"));
|
||||||
}
|
}
|
||||||
@@ -70,7 +69,7 @@ frappe.ui.form.on("Dunning", {
|
|||||||
get_query_filters: {
|
get_query_filters: {
|
||||||
docstatus: 1,
|
docstatus: 1,
|
||||||
status: "Overdue",
|
status: "Overdue",
|
||||||
company: frm.doc.company,
|
company: frm.doc.company
|
||||||
},
|
},
|
||||||
allow_child_item_selection: true,
|
allow_child_item_selection: true,
|
||||||
child_fieldname: "payment_schedule",
|
child_fieldname: "payment_schedule",
|
||||||
@@ -79,12 +78,9 @@ frappe.ui.form.on("Dunning", {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
frappe.dynamic_link = { doc: frm.doc, fieldname: "customer", doctype: "Customer" };
|
frappe.dynamic_link = { doc: frm.doc, fieldname: 'customer', doctype: 'Customer' };
|
||||||
|
|
||||||
frm.toggle_display(
|
frm.toggle_display("customer_name", (frm.doc.customer_name && frm.doc.customer_name !== frm.doc.customer));
|
||||||
"customer_name",
|
|
||||||
frm.doc.customer_name && frm.doc.customer_name !== frm.doc.customer
|
|
||||||
);
|
|
||||||
},
|
},
|
||||||
// When multiple companies are set up. in case company name is changed set default company address
|
// When multiple companies are set up. in case company name is changed set default company address
|
||||||
company: function (frm) {
|
company: function (frm) {
|
||||||
@@ -94,8 +90,8 @@ frappe.ui.form.on("Dunning", {
|
|||||||
args: { name: frm.doc.company, existing_address: frm.doc.company_address || "" },
|
args: { name: frm.doc.company, existing_address: frm.doc.company_address || "" },
|
||||||
debounce: 2000,
|
debounce: 2000,
|
||||||
callback: function (r) {
|
callback: function (r) {
|
||||||
frm.set_value("company_address", (r && r.message) || "");
|
frm.set_value("company_address", r && r.message || "");
|
||||||
},
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
if (frm.fields_dict.currency) {
|
if (frm.fields_dict.currency) {
|
||||||
@@ -129,16 +125,16 @@ frappe.ui.form.on("Dunning", {
|
|||||||
transaction_date: frm.doc.posting_date,
|
transaction_date: frm.doc.posting_date,
|
||||||
from_currency: frm.doc.currency,
|
from_currency: frm.doc.currency,
|
||||||
to_currency: company_currency,
|
to_currency: company_currency,
|
||||||
args: "for_selling",
|
args: "for_selling"
|
||||||
},
|
},
|
||||||
freeze: true,
|
freeze: true,
|
||||||
freeze_message: __("Fetching exchange rates ..."),
|
freeze_message: __("Fetching exchange rates ..."),
|
||||||
callback: function (r) {
|
callback: function(r) {
|
||||||
const exchange_rate = flt(r.message);
|
const exchange_rate = flt(r.message);
|
||||||
if (exchange_rate != frm.doc.conversion_rate) {
|
if (exchange_rate != frm.doc.conversion_rate) {
|
||||||
frm.set_value("conversion_rate", exchange_rate);
|
frm.set_value("conversion_rate", exchange_rate);
|
||||||
}
|
}
|
||||||
},
|
}
|
||||||
});
|
});
|
||||||
} else {
|
} else {
|
||||||
frm.trigger("conversion_rate");
|
frm.trigger("conversion_rate");
|
||||||
@@ -170,7 +166,8 @@ frappe.ui.form.on("Dunning", {
|
|||||||
get_dunning_letter_text: function (frm) {
|
get_dunning_letter_text: function (frm) {
|
||||||
if (frm.doc.dunning_type) {
|
if (frm.doc.dunning_type) {
|
||||||
frappe.call({
|
frappe.call({
|
||||||
method: "erpnext.accounts.doctype.dunning.dunning.get_dunning_letter_text",
|
method:
|
||||||
|
"erpnext.accounts.doctype.dunning.dunning.get_dunning_letter_text",
|
||||||
args: {
|
args: {
|
||||||
dunning_type: frm.doc.dunning_type,
|
dunning_type: frm.doc.dunning_type,
|
||||||
language: frm.doc.language,
|
language: frm.doc.language,
|
||||||
@@ -207,7 +204,10 @@ frappe.ui.form.on("Dunning", {
|
|||||||
calculate_overdue_days: function (frm) {
|
calculate_overdue_days: function (frm) {
|
||||||
frm.doc.overdue_payments.forEach((row) => {
|
frm.doc.overdue_payments.forEach((row) => {
|
||||||
if (frm.doc.posting_date && row.due_date) {
|
if (frm.doc.posting_date && row.due_date) {
|
||||||
const overdue_days = moment(frm.doc.posting_date).diff(row.due_date, "days");
|
const overdue_days = moment(frm.doc.posting_date).diff(
|
||||||
|
row.due_date,
|
||||||
|
"days"
|
||||||
|
);
|
||||||
frappe.model.set_value(row.doctype, row.name, "overdue_days", overdue_days);
|
frappe.model.set_value(row.doctype, row.name, "overdue_days", overdue_days);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
@@ -215,16 +215,15 @@ frappe.ui.form.on("Dunning", {
|
|||||||
calculate_interest: function (frm) {
|
calculate_interest: function (frm) {
|
||||||
frm.doc.overdue_payments.forEach((row) => {
|
frm.doc.overdue_payments.forEach((row) => {
|
||||||
const interest_per_day = frm.doc.rate_of_interest / 100 / 365;
|
const interest_per_day = frm.doc.rate_of_interest / 100 / 365;
|
||||||
const interest = flt(
|
const interest = flt((interest_per_day * row.overdue_days * row.outstanding), precision("interest", row));
|
||||||
interest_per_day * row.overdue_days * row.outstanding,
|
|
||||||
precision("interest", row)
|
|
||||||
);
|
|
||||||
frappe.model.set_value(row.doctype, row.name, "interest", interest);
|
frappe.model.set_value(row.doctype, row.name, "interest", interest);
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
calculate_totals: function (frm) {
|
calculate_totals: function (frm) {
|
||||||
const total_interest = frm.doc.overdue_payments.reduce((prev, cur) => prev + cur.interest, 0);
|
const total_interest = frm.doc.overdue_payments
|
||||||
const total_outstanding = frm.doc.overdue_payments.reduce((prev, cur) => prev + cur.outstanding, 0);
|
.reduce((prev, cur) => prev + cur.interest, 0);
|
||||||
|
const total_outstanding = frm.doc.overdue_payments
|
||||||
|
.reduce((prev, cur) => prev + cur.outstanding, 0);
|
||||||
const dunning_amount = total_interest + frm.doc.dunning_fee;
|
const dunning_amount = total_interest + frm.doc.dunning_fee;
|
||||||
const base_dunning_amount = dunning_amount * frm.doc.conversion_rate;
|
const base_dunning_amount = dunning_amount * frm.doc.conversion_rate;
|
||||||
const grand_total = total_outstanding + dunning_amount;
|
const grand_total = total_outstanding + dunning_amount;
|
||||||
@@ -241,7 +240,8 @@ frappe.ui.form.on("Dunning", {
|
|||||||
},
|
},
|
||||||
make_payment_entry: function (frm) {
|
make_payment_entry: function (frm) {
|
||||||
return frappe.call({
|
return frappe.call({
|
||||||
method: "erpnext.accounts.doctype.payment_entry.payment_entry.get_payment_entry",
|
method:
|
||||||
|
"erpnext.accounts.doctype.payment_entry.payment_entry.get_payment_entry",
|
||||||
args: {
|
args: {
|
||||||
dt: frm.doc.doctype,
|
dt: frm.doc.doctype,
|
||||||
dn: frm.doc.name,
|
dn: frm.doc.name,
|
||||||
@@ -257,5 +257,5 @@ frappe.ui.form.on("Dunning", {
|
|||||||
frappe.ui.form.on("Overdue Payment", {
|
frappe.ui.form.on("Overdue Payment", {
|
||||||
interest: function (frm) {
|
interest: function (frm) {
|
||||||
frm.trigger("calculate_totals");
|
frm.trigger("calculate_totals");
|
||||||
},
|
}
|
||||||
});
|
});
|
||||||
@@ -1,79 +1,75 @@
|
|||||||
// Copyright (c) 2018, Frappe Technologies Pvt. Ltd. and contributors
|
// Copyright (c) 2018, Frappe Technologies Pvt. Ltd. and contributors
|
||||||
// For license information, please see license.txt
|
// For license information, please see license.txt
|
||||||
|
|
||||||
frappe.ui.form.on("Exchange Rate Revaluation", {
|
frappe.ui.form.on('Exchange Rate Revaluation', {
|
||||||
setup: function (frm) {
|
setup: function(frm) {
|
||||||
frm.set_query("party_type", "accounts", function () {
|
frm.set_query("party_type", "accounts", function() {
|
||||||
return {
|
return {
|
||||||
filters: {
|
"filters": {
|
||||||
name: ["in", Object.keys(frappe.boot.party_account_types)],
|
"name": ["in", Object.keys(frappe.boot.party_account_types)],
|
||||||
},
|
}
|
||||||
};
|
};
|
||||||
});
|
});
|
||||||
frm.set_query("account", "accounts", function (doc) {
|
frm.set_query("account", "accounts", function(doc) {
|
||||||
return {
|
return {
|
||||||
filters: {
|
"filters": {
|
||||||
company: doc.company,
|
"company": doc.company
|
||||||
},
|
}
|
||||||
};
|
};
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
|
|
||||||
refresh: function (frm) {
|
refresh: function(frm) {
|
||||||
if (frm.doc.docstatus == 1) {
|
if(frm.doc.docstatus==1) {
|
||||||
frappe.call({
|
frappe.call({
|
||||||
method: "check_journal_entry_condition",
|
method: 'check_journal_entry_condition',
|
||||||
doc: frm.doc,
|
doc: frm.doc,
|
||||||
callback: function (r) {
|
callback: function(r) {
|
||||||
if (r.message) {
|
if (r.message) {
|
||||||
frm.add_custom_button(
|
frm.add_custom_button(__('Journal Entries'), function() {
|
||||||
__("Journal Entries"),
|
return frm.events.make_jv(frm);
|
||||||
function () {
|
}, __('Create'));
|
||||||
return frm.events.make_jv(frm);
|
|
||||||
},
|
|
||||||
__("Create")
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
},
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
validate_rounding_loss: function (frm) {
|
validate_rounding_loss: function(frm) {
|
||||||
let allowance = frm.doc.rounding_loss_allowance;
|
let allowance = frm.doc.rounding_loss_allowance;
|
||||||
if (!(allowance >= 0 && allowance < 1)) {
|
if (!(allowance >= 0 && allowance < 1)) {
|
||||||
frappe.throw(__("Rounding Loss Allowance should be between 0 and 1"));
|
frappe.throw(__("Rounding Loss Allowance should be between 0 and 1"));
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
rounding_loss_allowance: function (frm) {
|
rounding_loss_allowance: function(frm) {
|
||||||
frm.events.validate_rounding_loss(frm);
|
frm.events.validate_rounding_loss(frm);
|
||||||
},
|
},
|
||||||
|
|
||||||
validate: function (frm) {
|
validate: function(frm) {
|
||||||
frm.events.validate_rounding_loss(frm);
|
frm.events.validate_rounding_loss(frm);
|
||||||
},
|
},
|
||||||
|
|
||||||
get_entries: function (frm, account) {
|
get_entries: function(frm, account) {
|
||||||
frappe.call({
|
frappe.call({
|
||||||
method: "get_accounts_data",
|
method: "get_accounts_data",
|
||||||
doc: cur_frm.doc,
|
doc: cur_frm.doc,
|
||||||
account: account,
|
account: account,
|
||||||
callback: function (r) {
|
callback: function(r){
|
||||||
frappe.model.clear_table(frm.doc, "accounts");
|
frappe.model.clear_table(frm.doc, "accounts");
|
||||||
if (r.message) {
|
if(r.message) {
|
||||||
r.message.forEach((d) => {
|
r.message.forEach((d) => {
|
||||||
cur_frm.add_child("accounts", d);
|
cur_frm.add_child("accounts",d);
|
||||||
});
|
});
|
||||||
frm.events.get_total_gain_loss(frm);
|
frm.events.get_total_gain_loss(frm);
|
||||||
refresh_field("accounts");
|
refresh_field("accounts");
|
||||||
}
|
}
|
||||||
},
|
}
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
|
|
||||||
get_total_gain_loss: function (frm) {
|
get_total_gain_loss: function(frm) {
|
||||||
if (!(frm.doc.accounts && frm.doc.accounts.length)) return;
|
if(!(frm.doc.accounts && frm.doc.accounts.length)) return;
|
||||||
|
|
||||||
let total_gain_loss = 0;
|
let total_gain_loss = 0;
|
||||||
frm.doc.accounts.forEach((d) => {
|
frm.doc.accounts.forEach((d) => {
|
||||||
@@ -84,7 +80,7 @@ frappe.ui.form.on("Exchange Rate Revaluation", {
|
|||||||
frm.refresh_fields();
|
frm.refresh_fields();
|
||||||
},
|
},
|
||||||
|
|
||||||
make_jv: function (frm) {
|
make_jv : function(frm) {
|
||||||
let revaluation_journal = null;
|
let revaluation_journal = null;
|
||||||
let zero_balance_journal = null;
|
let zero_balance_journal = null;
|
||||||
frappe.call({
|
frappe.call({
|
||||||
@@ -92,68 +88,66 @@ frappe.ui.form.on("Exchange Rate Revaluation", {
|
|||||||
doc: frm.doc,
|
doc: frm.doc,
|
||||||
freeze: true,
|
freeze: true,
|
||||||
freeze_message: "Making Journal Entries...",
|
freeze_message: "Making Journal Entries...",
|
||||||
callback: function (r) {
|
callback: function(r){
|
||||||
if (r.message) {
|
if (r.message) {
|
||||||
let response = r.message;
|
let response = r.message;
|
||||||
if (response["revaluation_jv"] || response["zero_balance_jv"]) {
|
if(response['revaluation_jv'] || response['zero_balance_jv']) {
|
||||||
frappe.msgprint(__("Journals have been created"));
|
frappe.msgprint(__("Journals have been created"));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
}
|
||||||
});
|
});
|
||||||
},
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
frappe.ui.form.on("Exchange Rate Revaluation Account", {
|
frappe.ui.form.on("Exchange Rate Revaluation Account", {
|
||||||
new_exchange_rate: function (frm, cdt, cdn) {
|
new_exchange_rate: function(frm, cdt, cdn) {
|
||||||
var row = frappe.get_doc(cdt, cdn);
|
var row = frappe.get_doc(cdt, cdn);
|
||||||
row.new_balance_in_base_currency = flt(
|
row.new_balance_in_base_currency = flt(row.new_exchange_rate * flt(row.balance_in_account_currency),
|
||||||
row.new_exchange_rate * flt(row.balance_in_account_currency),
|
precision("new_balance_in_base_currency", row));
|
||||||
precision("new_balance_in_base_currency", row)
|
|
||||||
);
|
|
||||||
row.gain_loss = row.new_balance_in_base_currency - flt(row.balance_in_base_currency);
|
row.gain_loss = row.new_balance_in_base_currency - flt(row.balance_in_base_currency);
|
||||||
refresh_field("accounts");
|
refresh_field("accounts");
|
||||||
frm.events.get_total_gain_loss(frm);
|
frm.events.get_total_gain_loss(frm);
|
||||||
},
|
},
|
||||||
|
|
||||||
account: function (frm, cdt, cdn) {
|
account: function(frm, cdt, cdn) {
|
||||||
var row = locals[cdt][cdn];
|
var row = locals[cdt][cdn];
|
||||||
if (row.account) {
|
if (row.account) {
|
||||||
get_account_details(frm, cdt, cdn);
|
get_account_details(frm, cdt, cdn);
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
party: function (frm, cdt, cdn) {
|
party: function(frm, cdt, cdn) {
|
||||||
var row = locals[cdt][cdn];
|
var row = locals[cdt][cdn];
|
||||||
if (row.party && row.account) {
|
if (row.party && row.account) {
|
||||||
get_account_details(frm, cdt, cdn);
|
get_account_details(frm, cdt, cdn);
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
accounts_remove: function (frm) {
|
accounts_remove: function(frm) {
|
||||||
frm.events.get_total_gain_loss(frm);
|
frm.events.get_total_gain_loss(frm);
|
||||||
},
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
var get_account_details = function (frm, cdt, cdn) {
|
var get_account_details = function(frm, cdt, cdn) {
|
||||||
var row = frappe.get_doc(cdt, cdn);
|
var row = frappe.get_doc(cdt, cdn);
|
||||||
if (!frm.doc.company || !frm.doc.posting_date) {
|
if(!frm.doc.company || !frm.doc.posting_date) {
|
||||||
frappe.throw(__("Please select Company and Posting Date to getting entries"));
|
frappe.throw(__("Please select Company and Posting Date to getting entries"));
|
||||||
}
|
}
|
||||||
frappe.call({
|
frappe.call({
|
||||||
method: "erpnext.accounts.doctype.exchange_rate_revaluation.exchange_rate_revaluation.get_account_details",
|
method: "erpnext.accounts.doctype.exchange_rate_revaluation.exchange_rate_revaluation.get_account_details",
|
||||||
args: {
|
args:{
|
||||||
account: row.account,
|
account: row.account,
|
||||||
company: frm.doc.company,
|
company: frm.doc.company,
|
||||||
posting_date: frm.doc.posting_date,
|
posting_date: frm.doc.posting_date,
|
||||||
party_type: row.party_type,
|
party_type: row.party_type,
|
||||||
party: row.party,
|
party: row.party,
|
||||||
rounding_loss_allowance: frm.doc.rounding_loss_allowance,
|
rounding_loss_allowance: frm.doc.rounding_loss_allowance
|
||||||
},
|
},
|
||||||
callback: function (r) {
|
callback: function(r){
|
||||||
$.extend(row, r.message);
|
$.extend(row, r.message);
|
||||||
refresh_field("accounts");
|
refresh_field("accounts");
|
||||||
frm.events.get_total_gain_loss(frm);
|
frm.events.get_total_gain_loss(frm);
|
||||||
},
|
}
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -44,7 +44,7 @@ class ExchangeRateRevaluation(Document):
|
|||||||
self.set_total_gain_loss()
|
self.set_total_gain_loss()
|
||||||
|
|
||||||
def validate_rounding_loss_allowance(self):
|
def validate_rounding_loss_allowance(self):
|
||||||
if not (self.rounding_loss_allowance >= 0 and self.rounding_loss_allowance < 1):
|
if self.rounding_loss_allowance < 0 or self.rounding_loss_allowance >= 1:
|
||||||
frappe.throw(_("Rounding Loss Allowance should be between 0 and 1"))
|
frappe.throw(_("Rounding Loss Allowance should be between 0 and 1"))
|
||||||
|
|
||||||
def set_total_gain_loss(self):
|
def set_total_gain_loss(self):
|
||||||
|
|||||||
@@ -1,6 +1,8 @@
|
|||||||
// Copyright (c) 2018, Frappe Technologies Pvt. Ltd. and contributors
|
// Copyright (c) 2018, Frappe Technologies Pvt. Ltd. and contributors
|
||||||
// For license information, please see license.txt
|
// For license information, please see license.txt
|
||||||
|
|
||||||
frappe.ui.form.on("Finance Book", {
|
frappe.ui.form.on('Finance Book', {
|
||||||
refresh: function (frm) {},
|
refresh: function(frm) {
|
||||||
|
|
||||||
|
}
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -1,21 +1,17 @@
|
|||||||
// Copyright (c) 2015, Frappe Technologies Pvt. Ltd. and Contributors
|
// Copyright (c) 2015, Frappe Technologies Pvt. Ltd. and Contributors
|
||||||
// License: GNU General Public License v3. See license.txt
|
// License: GNU General Public License v3. See license.txt
|
||||||
|
|
||||||
frappe.ui.form.on("Fiscal Year", {
|
frappe.ui.form.on('Fiscal Year', {
|
||||||
onload: function (frm) {
|
onload: function(frm) {
|
||||||
if (frm.doc.__islocal) {
|
if(frm.doc.__islocal) {
|
||||||
frm.set_value(
|
frm.set_value("year_start_date",
|
||||||
"year_start_date",
|
frappe.datetime.add_days(frappe.defaults.get_default("year_end_date"), 1));
|
||||||
frappe.datetime.add_days(frappe.defaults.get_default("year_end_date"), 1)
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
year_start_date: function (frm) {
|
year_start_date: function(frm) {
|
||||||
if (!frm.doc.is_short_year) {
|
if (!frm.doc.is_short_year) {
|
||||||
let year_end_date = frappe.datetime.add_days(
|
let year_end_date =
|
||||||
frappe.datetime.add_months(frm.doc.year_start_date, 12),
|
frappe.datetime.add_days(frappe.datetime.add_months(frm.doc.year_start_date, 12), -1);
|
||||||
-1
|
|
||||||
);
|
|
||||||
frm.set_value("year_end_date", year_end_date);
|
frm.set_value("year_end_date", year_end_date);
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -3,7 +3,7 @@
|
|||||||
"allow_import": 1,
|
"allow_import": 1,
|
||||||
"autoname": "field:year",
|
"autoname": "field:year",
|
||||||
"creation": "2013-01-22 16:50:25",
|
"creation": "2013-01-22 16:50:25",
|
||||||
"description": "Represents a Financial Year. All accounting entries and other major transactions are tracked against the Fiscal Year.",
|
"description": "**Fiscal Year** represents a Financial Year. All accounting entries and other major transactions are tracked against **Fiscal Year**.",
|
||||||
"doctype": "DocType",
|
"doctype": "DocType",
|
||||||
"document_type": "Setup",
|
"document_type": "Setup",
|
||||||
"engine": "InnoDB",
|
"engine": "InnoDB",
|
||||||
@@ -82,11 +82,10 @@
|
|||||||
"icon": "fa fa-calendar",
|
"icon": "fa fa-calendar",
|
||||||
"idx": 1,
|
"idx": 1,
|
||||||
"links": [],
|
"links": [],
|
||||||
"modified": "2024-01-30 12:35:38.645968",
|
"modified": "2020-11-05 12:16:53.081573",
|
||||||
"modified_by": "Administrator",
|
"modified_by": "Administrator",
|
||||||
"module": "Accounts",
|
"module": "Accounts",
|
||||||
"name": "Fiscal Year",
|
"name": "Fiscal Year",
|
||||||
"naming_rule": "By fieldname",
|
|
||||||
"owner": "Administrator",
|
"owner": "Administrator",
|
||||||
"permissions": [
|
"permissions": [
|
||||||
{
|
{
|
||||||
@@ -123,6 +122,5 @@
|
|||||||
],
|
],
|
||||||
"show_name_in_global_search": 1,
|
"show_name_in_global_search": 1,
|
||||||
"sort_field": "name",
|
"sort_field": "name",
|
||||||
"sort_order": "DESC",
|
"sort_order": "DESC"
|
||||||
"states": []
|
|
||||||
}
|
}
|
||||||
@@ -39,7 +39,7 @@ def test_record_generator():
|
|||||||
]
|
]
|
||||||
|
|
||||||
start = 2012
|
start = 2012
|
||||||
end = now_datetime().year + 25
|
end = now_datetime().year + 5
|
||||||
for year in range(start, end):
|
for year in range(start, end):
|
||||||
test_records.append(
|
test_records.append(
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -1,8 +1,8 @@
|
|||||||
// Copyright (c) 2016, Frappe Technologies Pvt. Ltd. and contributors
|
// Copyright (c) 2016, Frappe Technologies Pvt. Ltd. and contributors
|
||||||
// For license information, please see license.txt
|
// For license information, please see license.txt
|
||||||
|
|
||||||
frappe.ui.form.on("GL Entry", {
|
frappe.ui.form.on('GL Entry', {
|
||||||
refresh: function (frm) {
|
refresh: function(frm) {
|
||||||
frm.page.btn_secondary.hide();
|
frm.page.btn_secondary.hide()
|
||||||
},
|
}
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -17,7 +17,9 @@
|
|||||||
"account_currency",
|
"account_currency",
|
||||||
"debit_in_account_currency",
|
"debit_in_account_currency",
|
||||||
"credit_in_account_currency",
|
"credit_in_account_currency",
|
||||||
|
"against_type",
|
||||||
"against",
|
"against",
|
||||||
|
"against_link",
|
||||||
"against_voucher_type",
|
"against_voucher_type",
|
||||||
"against_voucher",
|
"against_voucher",
|
||||||
"voucher_type",
|
"voucher_type",
|
||||||
@@ -129,6 +131,13 @@
|
|||||||
"label": "Credit Amount in Account Currency",
|
"label": "Credit Amount in Account Currency",
|
||||||
"options": "account_currency"
|
"options": "account_currency"
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "against_type",
|
||||||
|
"fieldtype": "Link",
|
||||||
|
"in_filter": 1,
|
||||||
|
"label": "Against Type",
|
||||||
|
"options": "DocType"
|
||||||
|
},
|
||||||
{
|
{
|
||||||
"fieldname": "against",
|
"fieldname": "against",
|
||||||
"fieldtype": "Text",
|
"fieldtype": "Text",
|
||||||
@@ -137,6 +146,13 @@
|
|||||||
"oldfieldname": "against",
|
"oldfieldname": "against",
|
||||||
"oldfieldtype": "Text"
|
"oldfieldtype": "Text"
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "against_link",
|
||||||
|
"fieldtype": "Dynamic Link",
|
||||||
|
"in_filter": 1,
|
||||||
|
"label": "Against",
|
||||||
|
"options": "against_type"
|
||||||
|
},
|
||||||
{
|
{
|
||||||
"fieldname": "against_voucher_type",
|
"fieldname": "against_voucher_type",
|
||||||
"fieldtype": "Link",
|
"fieldtype": "Link",
|
||||||
|
|||||||
@@ -13,9 +13,16 @@ import erpnext
|
|||||||
from erpnext.accounts.doctype.accounting_dimension.accounting_dimension import (
|
from erpnext.accounts.doctype.accounting_dimension.accounting_dimension import (
|
||||||
get_checks_for_pl_and_bs_accounts,
|
get_checks_for_pl_and_bs_accounts,
|
||||||
)
|
)
|
||||||
|
from erpnext.accounts.doctype.accounting_dimension_filter.accounting_dimension_filter import (
|
||||||
|
get_dimension_filter_map,
|
||||||
|
)
|
||||||
from erpnext.accounts.party import validate_party_frozen_disabled, validate_party_gle_currency
|
from erpnext.accounts.party import validate_party_frozen_disabled, validate_party_gle_currency
|
||||||
from erpnext.accounts.utils import get_account_currency, get_fiscal_year
|
from erpnext.accounts.utils import get_account_currency, get_fiscal_year
|
||||||
from erpnext.exceptions import InvalidAccountCurrency
|
from erpnext.exceptions import (
|
||||||
|
InvalidAccountCurrency,
|
||||||
|
InvalidAccountDimensionError,
|
||||||
|
MandatoryAccountDimensionError,
|
||||||
|
)
|
||||||
|
|
||||||
exclude_from_linked_with = True
|
exclude_from_linked_with = True
|
||||||
|
|
||||||
@@ -91,6 +98,7 @@ class GLEntry(Document):
|
|||||||
if not self.flags.from_repost and self.voucher_type != "Period Closing Voucher":
|
if not self.flags.from_repost and self.voucher_type != "Period Closing Voucher":
|
||||||
self.validate_account_details(adv_adj)
|
self.validate_account_details(adv_adj)
|
||||||
self.validate_dimensions_for_pl_and_bs()
|
self.validate_dimensions_for_pl_and_bs()
|
||||||
|
self.validate_allowed_dimensions()
|
||||||
validate_balance_type(self.account, adv_adj)
|
validate_balance_type(self.account, adv_adj)
|
||||||
validate_frozen_account(self.account, adv_adj)
|
validate_frozen_account(self.account, adv_adj)
|
||||||
|
|
||||||
@@ -200,6 +208,42 @@ class GLEntry(Document):
|
|||||||
)
|
)
|
||||||
)
|
)
|
||||||
|
|
||||||
|
def validate_allowed_dimensions(self):
|
||||||
|
dimension_filter_map = get_dimension_filter_map()
|
||||||
|
for key, value in dimension_filter_map.items():
|
||||||
|
dimension = key[0]
|
||||||
|
account = key[1]
|
||||||
|
|
||||||
|
if self.account == account:
|
||||||
|
if value["is_mandatory"] and not self.get(dimension):
|
||||||
|
frappe.throw(
|
||||||
|
_("{0} is mandatory for account {1}").format(
|
||||||
|
frappe.bold(frappe.unscrub(dimension)), frappe.bold(self.account)
|
||||||
|
),
|
||||||
|
MandatoryAccountDimensionError,
|
||||||
|
)
|
||||||
|
|
||||||
|
if value["allow_or_restrict"] == "Allow":
|
||||||
|
if self.get(dimension) and self.get(dimension) not in value["allowed_dimensions"]:
|
||||||
|
frappe.throw(
|
||||||
|
_("Invalid value {0} for {1} against account {2}").format(
|
||||||
|
frappe.bold(self.get(dimension)),
|
||||||
|
frappe.bold(frappe.unscrub(dimension)),
|
||||||
|
frappe.bold(self.account),
|
||||||
|
),
|
||||||
|
InvalidAccountDimensionError,
|
||||||
|
)
|
||||||
|
else:
|
||||||
|
if self.get(dimension) and self.get(dimension) in value["allowed_dimensions"]:
|
||||||
|
frappe.throw(
|
||||||
|
_("Invalid value {0} for {1} against account {2}").format(
|
||||||
|
frappe.bold(self.get(dimension)),
|
||||||
|
frappe.bold(frappe.unscrub(dimension)),
|
||||||
|
frappe.bold(self.account),
|
||||||
|
),
|
||||||
|
InvalidAccountDimensionError,
|
||||||
|
)
|
||||||
|
|
||||||
def check_pl_account(self):
|
def check_pl_account(self):
|
||||||
if (
|
if (
|
||||||
self.is_opening == "Yes"
|
self.is_opening == "Yes"
|
||||||
@@ -391,8 +435,8 @@ def update_outstanding_amt(
|
|||||||
def validate_frozen_account(account, adv_adj=None):
|
def validate_frozen_account(account, adv_adj=None):
|
||||||
frozen_account = frappe.get_cached_value("Account", account, "freeze_account")
|
frozen_account = frappe.get_cached_value("Account", account, "freeze_account")
|
||||||
if frozen_account == "Yes" and not adv_adj:
|
if frozen_account == "Yes" and not adv_adj:
|
||||||
frozen_accounts_modifier = frappe.db.get_value(
|
frozen_accounts_modifier = frappe.db.get_single_value(
|
||||||
"Accounts Settings", None, "frozen_accounts_modifier"
|
"Accounts Settings", "frozen_accounts_modifier"
|
||||||
)
|
)
|
||||||
|
|
||||||
if not frozen_accounts_modifier:
|
if not frozen_accounts_modifier:
|
||||||
|
|||||||
@@ -1,49 +1,47 @@
|
|||||||
// Copyright (c) 2019, Frappe Technologies Pvt. Ltd. and contributors
|
// Copyright (c) 2019, Frappe Technologies Pvt. Ltd. and contributors
|
||||||
// For license information, please see license.txt
|
// For license information, please see license.txt
|
||||||
|
|
||||||
frappe.ui.form.on("Invoice Discounting", {
|
frappe.ui.form.on('Invoice Discounting', {
|
||||||
setup: (frm) => {
|
setup: (frm) => {
|
||||||
frm.set_query("sales_invoice", "invoices", (doc) => {
|
frm.set_query("sales_invoice", "invoices", (doc) => {
|
||||||
return {
|
return {
|
||||||
filters: {
|
"filters": {
|
||||||
docstatus: 1,
|
"docstatus": 1,
|
||||||
company: doc.company,
|
"company": doc.company,
|
||||||
outstanding_amount: [">", 0],
|
"outstanding_amount": [">", 0]
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
||||||
frm.events.filter_accounts("bank_account", frm, [["account_type", "=", "Bank"]]);
|
frm.events.filter_accounts("bank_account", frm, [["account_type", "=", "Bank"]]);
|
||||||
frm.events.filter_accounts("bank_charges_account", frm, [["root_type", "=", "Expense"]]);
|
frm.events.filter_accounts("bank_charges_account", frm, [["root_type", "=", "Expense"]]);
|
||||||
frm.events.filter_accounts("short_term_loan", frm, [["root_type", "=", "Liability"]]);
|
frm.events.filter_accounts("short_term_loan", frm, [["root_type", "=", "Liability"]]);
|
||||||
frm.events.filter_accounts("accounts_receivable_discounted", frm, [
|
frm.events.filter_accounts("accounts_receivable_discounted", frm, [["account_type", "=", "Receivable"]]);
|
||||||
["account_type", "=", "Receivable"],
|
|
||||||
]);
|
|
||||||
frm.events.filter_accounts("accounts_receivable_credit", frm, [["account_type", "=", "Receivable"]]);
|
frm.events.filter_accounts("accounts_receivable_credit", frm, [["account_type", "=", "Receivable"]]);
|
||||||
frm.events.filter_accounts("accounts_receivable_unpaid", frm, [["account_type", "=", "Receivable"]]);
|
frm.events.filter_accounts("accounts_receivable_unpaid", frm, [["account_type", "=", "Receivable"]]);
|
||||||
|
|
||||||
},
|
},
|
||||||
|
|
||||||
filter_accounts: (fieldname, frm, addl_filters) => {
|
filter_accounts: (fieldname, frm, addl_filters) => {
|
||||||
let filters = [
|
let filters = [
|
||||||
["company", "=", frm.doc.company],
|
["company", "=", frm.doc.company],
|
||||||
["is_group", "=", 0],
|
["is_group", "=", 0]
|
||||||
];
|
];
|
||||||
if (addl_filters) {
|
if(addl_filters){
|
||||||
filters = $.merge(filters, addl_filters);
|
filters = $.merge(filters , addl_filters);
|
||||||
}
|
}
|
||||||
|
|
||||||
frm.set_query(fieldname, () => {
|
frm.set_query(fieldname, () => { return { "filters": filters }; });
|
||||||
return { filters: filters };
|
|
||||||
});
|
|
||||||
},
|
},
|
||||||
|
|
||||||
refresh_filters: (frm) => {
|
refresh_filters: (frm) =>{
|
||||||
let invoice_accounts = Object.keys(frm.doc.invoices).map(function (key) {
|
let invoice_accounts = Object.keys(frm.doc.invoices).map(function(key) {
|
||||||
return frm.doc.invoices[key].debit_to;
|
return frm.doc.invoices[key].debit_to;
|
||||||
});
|
});
|
||||||
let filters = [
|
let filters = [
|
||||||
["account_type", "=", "Receivable"],
|
["account_type", "=", "Receivable"],
|
||||||
["name", "not in", invoice_accounts],
|
["name", "not in", invoice_accounts]
|
||||||
];
|
];
|
||||||
frm.events.filter_accounts("accounts_receivable_credit", frm, filters);
|
frm.events.filter_accounts("accounts_receivable_credit", frm, filters);
|
||||||
frm.events.filter_accounts("accounts_receivable_discounted", frm, filters);
|
frm.events.filter_accounts("accounts_receivable_discounted", frm, filters);
|
||||||
@@ -54,19 +52,19 @@ frappe.ui.form.on("Invoice Discounting", {
|
|||||||
frm.events.show_general_ledger(frm);
|
frm.events.show_general_ledger(frm);
|
||||||
|
|
||||||
if (frm.doc.docstatus === 0) {
|
if (frm.doc.docstatus === 0) {
|
||||||
frm.add_custom_button(__("Get Invoices"), function () {
|
frm.add_custom_button(__('Get Invoices'), function() {
|
||||||
frm.events.get_invoices(frm);
|
frm.events.get_invoices(frm);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
if (frm.doc.docstatus === 1 && frm.doc.status !== "Settled") {
|
if (frm.doc.docstatus === 1 && frm.doc.status !== "Settled") {
|
||||||
if (frm.doc.status == "Sanctioned") {
|
if (frm.doc.status == "Sanctioned") {
|
||||||
frm.add_custom_button(__("Disburse Loan"), function () {
|
frm.add_custom_button(__('Disburse Loan'), function() {
|
||||||
frm.events.create_disbursement_entry(frm);
|
frm.events.create_disbursement_entry(frm);
|
||||||
}).addClass("btn-primary");
|
}).addClass("btn-primary");
|
||||||
}
|
}
|
||||||
if (frm.doc.status == "Disbursed") {
|
if (frm.doc.status == "Disbursed") {
|
||||||
frm.add_custom_button(__("Close Loan"), function () {
|
frm.add_custom_button(__('Close Loan'), function() {
|
||||||
frm.events.close_loan(frm);
|
frm.events.close_loan(frm);
|
||||||
}).addClass("btn-primary");
|
}).addClass("btn-primary");
|
||||||
}
|
}
|
||||||
@@ -94,121 +92,119 @@ frappe.ui.form.on("Invoice Discounting", {
|
|||||||
|
|
||||||
calculate_total_amount: (frm) => {
|
calculate_total_amount: (frm) => {
|
||||||
let total_amount = 0.0;
|
let total_amount = 0.0;
|
||||||
for (let row of frm.doc.invoices || []) {
|
for (let row of (frm.doc.invoices || [])) {
|
||||||
total_amount += flt(row.outstanding_amount);
|
total_amount += flt(row.outstanding_amount);
|
||||||
}
|
}
|
||||||
frm.set_value("total_amount", total_amount);
|
frm.set_value("total_amount", total_amount);
|
||||||
},
|
},
|
||||||
get_invoices: (frm) => {
|
get_invoices: (frm) => {
|
||||||
var d = new frappe.ui.Dialog({
|
var d = new frappe.ui.Dialog({
|
||||||
title: __("Get Invoices based on Filters"),
|
title: __('Get Invoices based on Filters'),
|
||||||
fields: [
|
fields: [
|
||||||
{
|
{
|
||||||
label: "Customer",
|
"label": "Customer",
|
||||||
fieldname: "customer",
|
"fieldname": "customer",
|
||||||
fieldtype: "Link",
|
"fieldtype": "Link",
|
||||||
options: "Customer",
|
"options": "Customer"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
label: "From Date",
|
"label": "From Date",
|
||||||
fieldname: "from_date",
|
"fieldname": "from_date",
|
||||||
fieldtype: "Date",
|
"fieldtype": "Date"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
label: "To Date",
|
"label": "To Date",
|
||||||
fieldname: "to_date",
|
"fieldname": "to_date",
|
||||||
fieldtype: "Date",
|
"fieldtype": "Date"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
fieldname: "col_break",
|
"fieldname": "col_break",
|
||||||
fieldtype: "Column Break",
|
"fieldtype": "Column Break",
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
label: "Min Amount",
|
"label": "Min Amount",
|
||||||
fieldname: "min_amount",
|
"fieldname": "min_amount",
|
||||||
fieldtype: "Currency",
|
"fieldtype": "Currency"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
label: "Max Amount",
|
"label": "Max Amount",
|
||||||
fieldname: "max_amount",
|
"fieldname": "max_amount",
|
||||||
fieldtype: "Currency",
|
"fieldtype": "Currency"
|
||||||
},
|
}
|
||||||
],
|
],
|
||||||
primary_action: function () {
|
primary_action: function() {
|
||||||
var data = d.get_values();
|
var data = d.get_values();
|
||||||
|
|
||||||
frappe.call({
|
frappe.call({
|
||||||
method: "erpnext.accounts.doctype.invoice_discounting.invoice_discounting.get_invoices",
|
method: "erpnext.accounts.doctype.invoice_discounting.invoice_discounting.get_invoices",
|
||||||
args: {
|
args: {
|
||||||
filters: data,
|
filters: data
|
||||||
},
|
},
|
||||||
callback: function (r) {
|
callback: function(r) {
|
||||||
if (!r.exc) {
|
if(!r.exc) {
|
||||||
d.hide();
|
d.hide();
|
||||||
$.each(r.message, function (i, v) {
|
$.each(r.message, function(i, v) {
|
||||||
frm.doc.invoices = frm.doc.invoices.filter((row) => row.sales_invoice);
|
frm.doc.invoices = frm.doc.invoices.filter(row => row.sales_invoice);
|
||||||
let row = frm.add_child("invoices");
|
let row = frm.add_child("invoices");
|
||||||
$.extend(row, v);
|
$.extend(row, v);
|
||||||
frm.events.refresh_filters(frm);
|
frm.events.refresh_filters(frm);
|
||||||
});
|
});
|
||||||
refresh_field("invoices");
|
refresh_field("invoices");
|
||||||
}
|
}
|
||||||
},
|
}
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
primary_action_label: __("Get Invocies"),
|
primary_action_label: __('Get Invocies')
|
||||||
});
|
});
|
||||||
d.show();
|
d.show();
|
||||||
},
|
},
|
||||||
|
|
||||||
create_disbursement_entry: (frm) => {
|
create_disbursement_entry: (frm) => {
|
||||||
frappe.call({
|
frappe.call({
|
||||||
method: "create_disbursement_entry",
|
method:"create_disbursement_entry",
|
||||||
doc: frm.doc,
|
doc: frm.doc,
|
||||||
callback: function (r) {
|
callback: function(r) {
|
||||||
if (!r.exc) {
|
if(!r.exc){
|
||||||
var doclist = frappe.model.sync(r.message);
|
var doclist = frappe.model.sync(r.message);
|
||||||
frappe.set_route("Form", doclist[0].doctype, doclist[0].name);
|
frappe.set_route("Form", doclist[0].doctype, doclist[0].name);
|
||||||
}
|
}
|
||||||
},
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
},
|
},
|
||||||
|
|
||||||
close_loan: (frm) => {
|
close_loan: (frm) => {
|
||||||
frappe.call({
|
frappe.call({
|
||||||
method: "close_loan",
|
method:"close_loan",
|
||||||
doc: frm.doc,
|
doc: frm.doc,
|
||||||
callback: function (r) {
|
callback: function(r) {
|
||||||
if (!r.exc) {
|
if(!r.exc){
|
||||||
var doclist = frappe.model.sync(r.message);
|
var doclist = frappe.model.sync(r.message);
|
||||||
frappe.set_route("Form", doclist[0].doctype, doclist[0].name);
|
frappe.set_route("Form", doclist[0].doctype, doclist[0].name);
|
||||||
}
|
}
|
||||||
},
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
},
|
},
|
||||||
|
|
||||||
show_general_ledger: (frm) => {
|
show_general_ledger: (frm) => {
|
||||||
if (frm.doc.docstatus > 0) {
|
if(frm.doc.docstatus > 0) {
|
||||||
cur_frm.add_custom_button(
|
cur_frm.add_custom_button(__('Accounting Ledger'), function() {
|
||||||
__("Accounting Ledger"),
|
frappe.route_options = {
|
||||||
function () {
|
voucher_no: frm.doc.name,
|
||||||
frappe.route_options = {
|
from_date: frm.doc.posting_date,
|
||||||
voucher_no: frm.doc.name,
|
to_date: moment(frm.doc.modified).format('YYYY-MM-DD'),
|
||||||
from_date: frm.doc.posting_date,
|
company: frm.doc.company,
|
||||||
to_date: moment(frm.doc.modified).format("YYYY-MM-DD"),
|
group_by: "Group by Voucher (Consolidated)",
|
||||||
company: frm.doc.company,
|
show_cancelled_entries: frm.doc.docstatus === 2
|
||||||
group_by: "Group by Voucher (Consolidated)",
|
};
|
||||||
show_cancelled_entries: frm.doc.docstatus === 2,
|
frappe.set_route("query-report", "General Ledger");
|
||||||
};
|
}, __("View"));
|
||||||
frappe.set_route("query-report", "General Ledger");
|
|
||||||
},
|
|
||||||
__("View")
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
},
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
frappe.ui.form.on("Discounted Invoice", {
|
frappe.ui.form.on('Discounted Invoice', {
|
||||||
sales_invoice: (frm) => {
|
sales_invoice: (frm) => {
|
||||||
frm.events.calculate_total_amount(frm);
|
frm.events.calculate_total_amount(frm);
|
||||||
frm.events.refresh_filters(frm);
|
frm.events.refresh_filters(frm);
|
||||||
@@ -216,5 +212,5 @@ frappe.ui.form.on("Discounted Invoice", {
|
|||||||
invoices_remove: (frm) => {
|
invoices_remove: (frm) => {
|
||||||
frm.events.calculate_total_amount(frm);
|
frm.events.calculate_total_amount(frm);
|
||||||
frm.events.refresh_filters(frm);
|
frm.events.refresh_filters(frm);
|
||||||
},
|
}
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -153,7 +153,9 @@ class InvoiceDiscounting(AccountsController):
|
|||||||
"account": inv.debit_to,
|
"account": inv.debit_to,
|
||||||
"party_type": "Customer",
|
"party_type": "Customer",
|
||||||
"party": d.customer,
|
"party": d.customer,
|
||||||
|
"against_type": "Account",
|
||||||
"against": self.accounts_receivable_credit,
|
"against": self.accounts_receivable_credit,
|
||||||
|
"against_link": self.accounts_receivable_credit,
|
||||||
"credit": outstanding_in_company_currency,
|
"credit": outstanding_in_company_currency,
|
||||||
"credit_in_account_currency": outstanding_in_company_currency
|
"credit_in_account_currency": outstanding_in_company_currency
|
||||||
if inv.party_account_currency == company_currency
|
if inv.party_account_currency == company_currency
|
||||||
@@ -173,7 +175,9 @@ class InvoiceDiscounting(AccountsController):
|
|||||||
"account": self.accounts_receivable_credit,
|
"account": self.accounts_receivable_credit,
|
||||||
"party_type": "Customer",
|
"party_type": "Customer",
|
||||||
"party": d.customer,
|
"party": d.customer,
|
||||||
|
"against_type": "Account",
|
||||||
"against": inv.debit_to,
|
"against": inv.debit_to,
|
||||||
|
"against_link": inv.debit_to,
|
||||||
"debit": outstanding_in_company_currency,
|
"debit": outstanding_in_company_currency,
|
||||||
"debit_in_account_currency": outstanding_in_company_currency
|
"debit_in_account_currency": outstanding_in_company_currency
|
||||||
if ar_credit_account_currency == company_currency
|
if ar_credit_account_currency == company_currency
|
||||||
|
|||||||
@@ -1,16 +1,21 @@
|
|||||||
frappe.listview_settings["Invoice Discounting"] = {
|
frappe.listview_settings['Invoice Discounting'] = {
|
||||||
add_fields: ["status"],
|
add_fields: ["status"],
|
||||||
get_indicator: function (doc) {
|
get_indicator: function(doc)
|
||||||
if (doc.status == "Draft") {
|
{
|
||||||
|
if(doc.status == "Draft") {
|
||||||
return [__("Draft"), "red", "status,=,Draft"];
|
return [__("Draft"), "red", "status,=,Draft"];
|
||||||
} else if (doc.status == "Sanctioned") {
|
}
|
||||||
|
else if(doc.status == "Sanctioned") {
|
||||||
return [__("Sanctioned"), "green", "status,=,Sanctioned"];
|
return [__("Sanctioned"), "green", "status,=,Sanctioned"];
|
||||||
} else if (doc.status == "Disbursed") {
|
}
|
||||||
|
else if(doc.status == "Disbursed") {
|
||||||
return [__("Disbursed"), "blue", "status,=,Disbursed"];
|
return [__("Disbursed"), "blue", "status,=,Disbursed"];
|
||||||
} else if (doc.status == "Settled") {
|
}
|
||||||
|
else if(doc.status == "Settled") {
|
||||||
return [__("Settled"), "orange", "status,=,Settled"];
|
return [__("Settled"), "orange", "status,=,Settled"];
|
||||||
} else if (doc.status == "Canceled") {
|
}
|
||||||
|
else if(doc.status == "Canceled") {
|
||||||
return [__("Canceled"), "red", "status,=,Canceled"];
|
return [__("Canceled"), "red", "status,=,Canceled"];
|
||||||
}
|
}
|
||||||
},
|
}
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -1,49 +1,27 @@
|
|||||||
// Copyright (c) 2018, Frappe Technologies Pvt. Ltd. and contributors
|
// Copyright (c) 2018, Frappe Technologies Pvt. Ltd. and contributors
|
||||||
// For license information, please see license.txt
|
// For license information, please see license.txt
|
||||||
|
|
||||||
frappe.ui.form.on("Item Tax Template", {
|
frappe.ui.form.on('Item Tax Template', {
|
||||||
setup: function (frm) {
|
setup: function(frm) {
|
||||||
frm.set_query("tax_type", "taxes", function (doc) {
|
frm.set_query("tax_type", "taxes", function(doc) {
|
||||||
return {
|
return {
|
||||||
filters: [
|
filters: [
|
||||||
["Account", "company", "=", frm.doc.company],
|
['Account', 'company', '=', frm.doc.company],
|
||||||
["Account", "is_group", "=", 0],
|
['Account', 'is_group', '=', 0],
|
||||||
[
|
['Account', 'account_type', 'in', ['Tax', 'Chargeable', 'Income Account', 'Expense Account', 'Expenses Included In Valuation']]
|
||||||
"Account",
|
]
|
||||||
"account_type",
|
}
|
||||||
"in",
|
|
||||||
[
|
|
||||||
"Tax",
|
|
||||||
"Chargeable",
|
|
||||||
"Income Account",
|
|
||||||
"Expense Account",
|
|
||||||
"Expenses Included In Valuation",
|
|
||||||
],
|
|
||||||
],
|
|
||||||
],
|
|
||||||
};
|
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
company: function (frm) {
|
company: function (frm) {
|
||||||
frm.set_query("tax_type", "taxes", function (doc) {
|
frm.set_query("tax_type", "taxes", function(doc) {
|
||||||
return {
|
return {
|
||||||
filters: [
|
filters: [
|
||||||
["Account", "company", "=", frm.doc.company],
|
['Account', 'company', '=', frm.doc.company],
|
||||||
["Account", "is_group", "=", 0],
|
['Account', 'is_group', '=', 0],
|
||||||
[
|
['Account', 'account_type', 'in', ['Tax', 'Chargeable', 'Income Account', 'Expense Account', 'Expenses Included In Valuation']]
|
||||||
"Account",
|
]
|
||||||
"account_type",
|
}
|
||||||
"in",
|
|
||||||
[
|
|
||||||
"Tax",
|
|
||||||
"Chargeable",
|
|
||||||
"Income Account",
|
|
||||||
"Expense Account",
|
|
||||||
"Expenses Included In Valuation",
|
|
||||||
],
|
|
||||||
],
|
|
||||||
],
|
|
||||||
};
|
|
||||||
});
|
});
|
||||||
},
|
}
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -4,82 +4,39 @@
|
|||||||
frappe.provide("erpnext.accounts");
|
frappe.provide("erpnext.accounts");
|
||||||
frappe.provide("erpnext.journal_entry");
|
frappe.provide("erpnext.journal_entry");
|
||||||
|
|
||||||
|
|
||||||
frappe.ui.form.on("Journal Entry", {
|
frappe.ui.form.on("Journal Entry", {
|
||||||
setup: function (frm) {
|
setup: function(frm) {
|
||||||
frm.add_fetch("bank_account", "account", "account");
|
frm.add_fetch("bank_account", "account", "account");
|
||||||
frm.ignore_doctypes_on_cancel_all = [
|
frm.ignore_doctypes_on_cancel_all = ['Sales Invoice', 'Purchase Invoice', 'Journal Entry', "Repost Payment Ledger", 'Asset', 'Asset Movement', 'Asset Depreciation Schedule', "Repost Accounting Ledger", "Unreconcile Payment", "Unreconcile Payment Entries"];
|
||||||
"Sales Invoice",
|
|
||||||
"Purchase Invoice",
|
|
||||||
"Journal Entry",
|
|
||||||
"Repost Payment Ledger",
|
|
||||||
"Asset",
|
|
||||||
"Asset Movement",
|
|
||||||
"Asset Depreciation Schedule",
|
|
||||||
"Repost Accounting Ledger",
|
|
||||||
"Unreconcile Payment",
|
|
||||||
"Unreconcile Payment Entries",
|
|
||||||
"Bank Transaction",
|
|
||||||
];
|
|
||||||
},
|
},
|
||||||
|
|
||||||
refresh: function (frm) {
|
refresh: function(frm) {
|
||||||
erpnext.toggle_naming_series();
|
erpnext.toggle_naming_series();
|
||||||
|
|
||||||
if (frm.doc.repost_required && frm.doc.docstatus === 1) {
|
if(frm.doc.docstatus > 0) {
|
||||||
frm.set_intro(
|
frm.add_custom_button(__('Ledger'), function() {
|
||||||
__(
|
frappe.route_options = {
|
||||||
"Accounting entries for this Journal Entry need to be reposted. Please click on 'Repost' button to update."
|
"voucher_no": frm.doc.name,
|
||||||
)
|
"from_date": frm.doc.posting_date,
|
||||||
);
|
"to_date": moment(frm.doc.modified).format('YYYY-MM-DD'),
|
||||||
frm.add_custom_button(__("Repost Accounting Entries"), () => {
|
"company": frm.doc.company,
|
||||||
frm.call({
|
"finance_book": frm.doc.finance_book,
|
||||||
doc: frm.doc,
|
"group_by": '',
|
||||||
method: "repost_accounting_entries",
|
"show_cancelled_entries": frm.doc.docstatus === 2
|
||||||
freeze: true,
|
};
|
||||||
freeze_message: __("Reposting..."),
|
frappe.set_route("query-report", "General Ledger");
|
||||||
callback: (r) => {
|
}, __('View'));
|
||||||
if (!r.exc) {
|
|
||||||
frappe.msgprint(__("Accounting Entries are reposted."));
|
|
||||||
frm.refresh();
|
|
||||||
}
|
|
||||||
},
|
|
||||||
});
|
|
||||||
})
|
|
||||||
.removeClass("btn-default")
|
|
||||||
.addClass("btn-warning");
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (frm.doc.docstatus > 0) {
|
if(frm.doc.docstatus==1) {
|
||||||
frm.add_custom_button(
|
frm.add_custom_button(__('Reverse Journal Entry'), function() {
|
||||||
__("Ledger"),
|
return erpnext.journal_entry.reverse_journal_entry(frm);
|
||||||
function () {
|
}, __('Actions'));
|
||||||
frappe.route_options = {
|
|
||||||
voucher_no: frm.doc.name,
|
|
||||||
from_date: frm.doc.posting_date,
|
|
||||||
to_date: moment(frm.doc.modified).format("YYYY-MM-DD"),
|
|
||||||
company: frm.doc.company,
|
|
||||||
finance_book: frm.doc.finance_book,
|
|
||||||
group_by: "",
|
|
||||||
show_cancelled_entries: frm.doc.docstatus === 2,
|
|
||||||
};
|
|
||||||
frappe.set_route("query-report", "General Ledger");
|
|
||||||
},
|
|
||||||
__("View")
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (frm.doc.docstatus == 1) {
|
|
||||||
frm.add_custom_button(
|
|
||||||
__("Reverse Journal Entry"),
|
|
||||||
function () {
|
|
||||||
return erpnext.journal_entry.reverse_journal_entry(frm);
|
|
||||||
},
|
|
||||||
__("Actions")
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (frm.doc.__islocal) {
|
if (frm.doc.__islocal) {
|
||||||
frm.add_custom_button(__("Quick Entry"), function () {
|
frm.add_custom_button(__('Quick Entry'), function() {
|
||||||
return erpnext.journal_entry.quick_entry(frm);
|
return erpnext.journal_entry.quick_entry(frm);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
@@ -87,63 +44,52 @@ frappe.ui.form.on("Journal Entry", {
|
|||||||
// hide /unhide fields based on currency
|
// hide /unhide fields based on currency
|
||||||
erpnext.journal_entry.toggle_fields_based_on_currency(frm);
|
erpnext.journal_entry.toggle_fields_based_on_currency(frm);
|
||||||
|
|
||||||
if (
|
if ((frm.doc.voucher_type == "Inter Company Journal Entry") && (frm.doc.docstatus == 1) && (!frm.doc.inter_company_journal_entry_reference)) {
|
||||||
frm.doc.voucher_type == "Inter Company Journal Entry" &&
|
frm.add_custom_button(__("Create Inter Company Journal Entry"),
|
||||||
frm.doc.docstatus == 1 &&
|
function() {
|
||||||
!frm.doc.inter_company_journal_entry_reference
|
|
||||||
) {
|
|
||||||
frm.add_custom_button(
|
|
||||||
__("Create Inter Company Journal Entry"),
|
|
||||||
function () {
|
|
||||||
frm.trigger("make_inter_company_journal_entry");
|
frm.trigger("make_inter_company_journal_entry");
|
||||||
},
|
}, __('Make'));
|
||||||
__("Make")
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
erpnext.accounts.unreconcile_payment.add_unreconcile_btn(frm);
|
erpnext.accounts.unreconcile_payment.add_unreconcile_btn(frm);
|
||||||
},
|
},
|
||||||
before_save: function (frm) {
|
before_save: function(frm) {
|
||||||
if (frm.doc.docstatus == 0 && !frm.doc.is_system_generated) {
|
if ((frm.doc.docstatus == 0) && (!frm.doc.is_system_generated)) {
|
||||||
let payment_entry_references = frm.doc.accounts.filter(
|
let payment_entry_references = frm.doc.accounts.filter(elem => (elem.reference_type == "Payment Entry"));
|
||||||
(elem) => elem.reference_type == "Payment Entry"
|
|
||||||
);
|
|
||||||
if (payment_entry_references.length > 0) {
|
if (payment_entry_references.length > 0) {
|
||||||
let rows = payment_entry_references.map((x) => "#" + x.idx);
|
let rows = payment_entry_references.map(x => "#"+x.idx);
|
||||||
frappe.throw(
|
frappe.throw(__("Rows: {0} have 'Payment Entry' as reference_type. This should not be set manually.", [frappe.utils.comma_and(rows)]));
|
||||||
__("Rows: {0} have 'Payment Entry' as reference_type. This should not be set manually.", [
|
|
||||||
frappe.utils.comma_and(rows),
|
|
||||||
])
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
make_inter_company_journal_entry: function (frm) {
|
make_inter_company_journal_entry: function(frm) {
|
||||||
var d = new frappe.ui.Dialog({
|
var d = new frappe.ui.Dialog({
|
||||||
title: __("Select Company"),
|
title: __("Select Company"),
|
||||||
fields: [
|
fields: [
|
||||||
{
|
{
|
||||||
fieldname: "company",
|
'fieldname': 'company',
|
||||||
fieldtype: "Link",
|
'fieldtype': 'Link',
|
||||||
label: __("Company"),
|
'label': __('Company'),
|
||||||
options: "Company",
|
'options': 'Company',
|
||||||
get_query: function () {
|
"get_query": function () {
|
||||||
return {
|
return {
|
||||||
filters: [["Company", "name", "!=", frm.doc.company]],
|
filters: [
|
||||||
|
["Company", "name", "!=", frm.doc.company]
|
||||||
|
]
|
||||||
};
|
};
|
||||||
},
|
},
|
||||||
reqd: 1,
|
'reqd': 1
|
||||||
},
|
}
|
||||||
],
|
],
|
||||||
});
|
});
|
||||||
d.set_primary_action(__("Create"), function () {
|
d.set_primary_action(__('Create'), function() {
|
||||||
d.hide();
|
d.hide();
|
||||||
var args = d.get_values();
|
var args = d.get_values();
|
||||||
frappe.call({
|
frappe.call({
|
||||||
args: {
|
args: {
|
||||||
name: frm.doc.name,
|
"name": frm.doc.name,
|
||||||
voucher_type: frm.doc.voucher_type,
|
"voucher_type": frm.doc.voucher_type,
|
||||||
company: args.company,
|
"company": args.company
|
||||||
},
|
},
|
||||||
method: "erpnext.accounts.doctype.journal_entry.journal_entry.make_inter_company_journal_entry",
|
method: "erpnext.accounts.doctype.journal_entry.journal_entry.make_inter_company_journal_entry",
|
||||||
callback: function (r) {
|
callback: function (r) {
|
||||||
@@ -151,106 +97,103 @@ frappe.ui.form.on("Journal Entry", {
|
|||||||
var doc = frappe.model.sync(r.message)[0];
|
var doc = frappe.model.sync(r.message)[0];
|
||||||
frappe.set_route("Form", doc.doctype, doc.name);
|
frappe.set_route("Form", doc.doctype, doc.name);
|
||||||
}
|
}
|
||||||
},
|
}
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
d.show();
|
d.show();
|
||||||
},
|
},
|
||||||
|
|
||||||
multi_currency: function (frm) {
|
multi_currency: function(frm) {
|
||||||
erpnext.journal_entry.toggle_fields_based_on_currency(frm);
|
erpnext.journal_entry.toggle_fields_based_on_currency(frm);
|
||||||
},
|
},
|
||||||
|
|
||||||
posting_date: function (frm) {
|
posting_date: function(frm) {
|
||||||
if (!frm.doc.multi_currency || !frm.doc.posting_date) return;
|
if(!frm.doc.multi_currency || !frm.doc.posting_date) return;
|
||||||
|
|
||||||
$.each(frm.doc.accounts || [], function (i, row) {
|
$.each(frm.doc.accounts || [], function(i, row) {
|
||||||
erpnext.journal_entry.set_exchange_rate(frm, row.doctype, row.name);
|
erpnext.journal_entry.set_exchange_rate(frm, row.doctype, row.name);
|
||||||
});
|
})
|
||||||
},
|
},
|
||||||
|
|
||||||
company: function (frm) {
|
company: function(frm) {
|
||||||
frappe.call({
|
frappe.call({
|
||||||
method: "frappe.client.get_value",
|
method: "frappe.client.get_value",
|
||||||
args: {
|
args: {
|
||||||
doctype: "Company",
|
doctype: "Company",
|
||||||
filters: { name: frm.doc.company },
|
filters: {"name": frm.doc.company},
|
||||||
fieldname: "cost_center",
|
fieldname: "cost_center"
|
||||||
},
|
},
|
||||||
callback: function (r) {
|
callback: function(r){
|
||||||
if (r.message) {
|
if(r.message){
|
||||||
$.each(frm.doc.accounts || [], function (i, jvd) {
|
$.each(frm.doc.accounts || [], function(i, jvd) {
|
||||||
frappe.model.set_value(jvd.doctype, jvd.name, "cost_center", r.message.cost_center);
|
frappe.model.set_value(jvd.doctype, jvd.name, "cost_center", r.message.cost_center);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
},
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
erpnext.accounts.dimensions.update_dimension(frm, frm.doctype);
|
erpnext.accounts.dimensions.update_dimension(frm, frm.doctype);
|
||||||
},
|
},
|
||||||
|
|
||||||
voucher_type: function (frm) {
|
voucher_type: function(frm){
|
||||||
if (!frm.doc.company) return null;
|
|
||||||
|
|
||||||
if (
|
if(!frm.doc.company) return null;
|
||||||
!(frm.doc.accounts || []).length ||
|
|
||||||
((frm.doc.accounts || []).length === 1 && !frm.doc.accounts[0].account)
|
if((!(frm.doc.accounts || []).length) || ((frm.doc.accounts || []).length === 1 && !frm.doc.accounts[0].account)) {
|
||||||
) {
|
if(in_list(["Bank Entry", "Cash Entry"], frm.doc.voucher_type)) {
|
||||||
if (in_list(["Bank Entry", "Cash Entry"], frm.doc.voucher_type)) {
|
|
||||||
return frappe.call({
|
return frappe.call({
|
||||||
type: "GET",
|
type: "GET",
|
||||||
method: "erpnext.accounts.doctype.journal_entry.journal_entry.get_default_bank_cash_account",
|
method: "erpnext.accounts.doctype.journal_entry.journal_entry.get_default_bank_cash_account",
|
||||||
args: {
|
args: {
|
||||||
account_type:
|
"account_type": (frm.doc.voucher_type=="Bank Entry" ?
|
||||||
frm.doc.voucher_type == "Bank Entry"
|
"Bank" : (frm.doc.voucher_type=="Cash Entry" ? "Cash" : null)),
|
||||||
? "Bank"
|
"company": frm.doc.company
|
||||||
: frm.doc.voucher_type == "Cash Entry"
|
|
||||||
? "Cash"
|
|
||||||
: null,
|
|
||||||
company: frm.doc.company,
|
|
||||||
},
|
},
|
||||||
callback: function (r) {
|
callback: function(r) {
|
||||||
if (r.message) {
|
if(r.message) {
|
||||||
// If default company bank account not set
|
// If default company bank account not set
|
||||||
if (!$.isEmptyObject(r.message)) {
|
if(!$.isEmptyObject(r.message)){
|
||||||
update_jv_details(frm.doc, [r.message]);
|
update_jv_details(frm.doc, [r.message]);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
from_template: function (frm) {
|
from_template: function(frm){
|
||||||
if (frm.doc.from_template) {
|
if (frm.doc.from_template){
|
||||||
frappe.db.get_doc("Journal Entry Template", frm.doc.from_template).then((doc) => {
|
frappe.db.get_doc("Journal Entry Template", frm.doc.from_template)
|
||||||
frappe.model.clear_table(frm.doc, "accounts");
|
.then((doc) => {
|
||||||
frm.set_value({
|
frappe.model.clear_table(frm.doc, "accounts");
|
||||||
company: doc.company,
|
frm.set_value({
|
||||||
voucher_type: doc.voucher_type,
|
"company": doc.company,
|
||||||
naming_series: doc.naming_series,
|
"voucher_type": doc.voucher_type,
|
||||||
is_opening: doc.is_opening,
|
"naming_series": doc.naming_series,
|
||||||
multi_currency: doc.multi_currency,
|
"is_opening": doc.is_opening,
|
||||||
|
"multi_currency": doc.multi_currency
|
||||||
|
})
|
||||||
|
update_jv_details(frm.doc, doc.accounts);
|
||||||
});
|
});
|
||||||
update_jv_details(frm.doc, doc.accounts);
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
},
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
var update_jv_details = function (doc, r) {
|
var update_jv_details = function(doc, r) {
|
||||||
$.each(r, function (i, d) {
|
$.each(r, function(i, d) {
|
||||||
var row = frappe.model.add_child(doc, "Journal Entry Account", "accounts");
|
var row = frappe.model.add_child(doc, "Journal Entry Account", "accounts");
|
||||||
frappe.model.set_value(row.doctype, row.name, "account", d.account);
|
frappe.model.set_value(row.doctype, row.name, "account", d.account)
|
||||||
|
frappe.model.set_value(row.doctype, row.name, "balance", d.balance)
|
||||||
});
|
});
|
||||||
refresh_field("accounts");
|
refresh_field("accounts");
|
||||||
};
|
}
|
||||||
|
|
||||||
erpnext.accounts.JournalEntry = class JournalEntry extends frappe.ui.form.Controller {
|
erpnext.accounts.JournalEntry = class JournalEntry extends frappe.ui.form.Controller {
|
||||||
onload() {
|
onload() {
|
||||||
this.load_defaults();
|
this.load_defaults();
|
||||||
this.setup_queries();
|
this.setup_queries();
|
||||||
|
this.setup_balance_formatter();
|
||||||
erpnext.accounts.dimensions.setup_dimension_filters(this.frm, this.frm.doctype);
|
erpnext.accounts.dimensions.setup_dimension_filters(this.frm, this.frm.doctype);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -260,67 +203,79 @@ erpnext.accounts.JournalEntry = class JournalEntry extends frappe.ui.form.Contro
|
|||||||
|
|
||||||
load_defaults() {
|
load_defaults() {
|
||||||
//this.frm.show_print_first = true;
|
//this.frm.show_print_first = true;
|
||||||
if (this.frm.doc.__islocal && this.frm.doc.company) {
|
if(this.frm.doc.__islocal && this.frm.doc.company) {
|
||||||
frappe.model.set_default_values(this.frm.doc);
|
frappe.model.set_default_values(this.frm.doc);
|
||||||
$.each(this.frm.doc.accounts || [], function (i, jvd) {
|
$.each(this.frm.doc.accounts || [], function(i, jvd) {
|
||||||
frappe.model.set_default_values(jvd);
|
frappe.model.set_default_values(jvd);
|
||||||
});
|
});
|
||||||
var posting_date = this.frm.doc.posting_date;
|
var posting_date = this.frm.doc.posting_date;
|
||||||
if (!this.frm.doc.amended_from)
|
if(!this.frm.doc.amended_from) this.frm.set_value('posting_date', posting_date || frappe.datetime.get_today());
|
||||||
this.frm.set_value("posting_date", posting_date || frappe.datetime.get_today());
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
setup_queries() {
|
setup_queries() {
|
||||||
var me = this;
|
var me = this;
|
||||||
|
|
||||||
me.frm.set_query("account", "accounts", function (doc, cdt, cdn) {
|
me.frm.set_query("account", "accounts", function(doc, cdt, cdn) {
|
||||||
return erpnext.journal_entry.account_query(me.frm);
|
return erpnext.journal_entry.account_query(me.frm);
|
||||||
});
|
});
|
||||||
|
|
||||||
me.frm.set_query("party_type", "accounts", function (doc, cdt, cdn) {
|
me.frm.set_query("against_account_link", "accounts", function(doc, cdt, cdn) {
|
||||||
|
return erpnext.journal_entry.against_account_query(me.frm);
|
||||||
|
});
|
||||||
|
|
||||||
|
me.frm.set_query("against_type", "accounts", function(){
|
||||||
|
return {
|
||||||
|
query: "erpnext.accounts.doctype.journal_entry.journal_entry.get_against_type",
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
me.frm.set_query("party_type", "accounts", function(doc, cdt, cdn) {
|
||||||
const row = locals[cdt][cdn];
|
const row = locals[cdt][cdn];
|
||||||
|
|
||||||
return {
|
return {
|
||||||
query: "erpnext.setup.doctype.party_type.party_type.get_party_type",
|
query: "erpnext.setup.doctype.party_type.party_type.get_party_type",
|
||||||
filters: {
|
filters: {
|
||||||
account: row.account,
|
'account': row.account
|
||||||
},
|
}
|
||||||
};
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
me.frm.set_query("reference_name", "accounts", function (doc, cdt, cdn) {
|
me.frm.set_query("reference_name", "accounts", function(doc, cdt, cdn) {
|
||||||
var jvd = frappe.get_doc(cdt, cdn);
|
var jvd = frappe.get_doc(cdt, cdn);
|
||||||
|
|
||||||
// journal entry
|
// journal entry
|
||||||
if (jvd.reference_type === "Journal Entry") {
|
if(jvd.reference_type==="Journal Entry") {
|
||||||
frappe.model.validate_missing(jvd, "account");
|
frappe.model.validate_missing(jvd, "account");
|
||||||
return {
|
return {
|
||||||
query: "erpnext.accounts.doctype.journal_entry.journal_entry.get_against_jv",
|
query: "erpnext.accounts.doctype.journal_entry.journal_entry.get_against_jv",
|
||||||
filters: {
|
filters: {
|
||||||
account: jvd.account,
|
account: jvd.account,
|
||||||
party: jvd.party,
|
party: jvd.party
|
||||||
},
|
}
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
var out = {
|
var out = {
|
||||||
filters: [[jvd.reference_type, "docstatus", "=", 1]],
|
filters: [
|
||||||
|
[jvd.reference_type, "docstatus", "=", 1]
|
||||||
|
]
|
||||||
};
|
};
|
||||||
|
|
||||||
if (in_list(["Sales Invoice", "Purchase Invoice"], jvd.reference_type)) {
|
if(in_list(["Sales Invoice", "Purchase Invoice"], jvd.reference_type)) {
|
||||||
out.filters.push([jvd.reference_type, "outstanding_amount", "!=", 0]);
|
out.filters.push([jvd.reference_type, "outstanding_amount", "!=", 0]);
|
||||||
// Filter by cost center
|
// Filter by cost center
|
||||||
if (jvd.cost_center) {
|
if(jvd.cost_center) {
|
||||||
out.filters.push([jvd.reference_type, "cost_center", "in", ["", jvd.cost_center]]);
|
out.filters.push([jvd.reference_type, "cost_center", "in", ["", jvd.cost_center]]);
|
||||||
}
|
}
|
||||||
// account filter
|
// account filter
|
||||||
frappe.model.validate_missing(jvd, "account");
|
frappe.model.validate_missing(jvd, "account");
|
||||||
var party_account_field = jvd.reference_type === "Sales Invoice" ? "debit_to" : "credit_to";
|
var party_account_field = jvd.reference_type==="Sales Invoice" ? "debit_to": "credit_to";
|
||||||
out.filters.push([jvd.reference_type, party_account_field, "=", jvd.account]);
|
out.filters.push([jvd.reference_type, party_account_field, "=", jvd.account]);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (in_list(["Sales Order", "Purchase Order"], jvd.reference_type)) {
|
if(in_list(["Sales Order", "Purchase Order"], jvd.reference_type)) {
|
||||||
// party_type and party mandatory
|
// party_type and party mandatory
|
||||||
frappe.model.validate_missing(jvd, "party_type");
|
frappe.model.validate_missing(jvd, "party_type");
|
||||||
frappe.model.validate_missing(jvd, "party");
|
frappe.model.validate_missing(jvd, "party");
|
||||||
@@ -328,11 +283,11 @@ erpnext.accounts.JournalEntry = class JournalEntry extends frappe.ui.form.Contro
|
|||||||
out.filters.push([jvd.reference_type, "per_billed", "<", 100]);
|
out.filters.push([jvd.reference_type, "per_billed", "<", 100]);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (jvd.party_type && jvd.party) {
|
if(jvd.party_type && jvd.party) {
|
||||||
let party_field = "";
|
let party_field = "";
|
||||||
if (jvd.reference_type.indexOf("Sales") === 0) {
|
if(jvd.reference_type.indexOf("Sales")===0) {
|
||||||
party_field = "customer";
|
party_field = "customer";
|
||||||
} else if (jvd.reference_type.indexOf("Purchase") === 0) {
|
} else if (jvd.reference_type.indexOf("Purchase")===0) {
|
||||||
party_field = "supplier";
|
party_field = "supplier";
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -343,49 +298,64 @@ erpnext.accounts.JournalEntry = class JournalEntry extends frappe.ui.form.Contro
|
|||||||
|
|
||||||
return out;
|
return out;
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
setup_balance_formatter() {
|
||||||
|
const formatter = function(value, df, options, doc) {
|
||||||
|
var currency = frappe.meta.get_field_currency(df, doc);
|
||||||
|
var dr_or_cr = value ? ('<label>' + (value > 0.0 ? __("Dr") : __("Cr")) + '</label>') : "";
|
||||||
|
return "<div style='text-align: right'>"
|
||||||
|
+ ((value==null || value==="") ? "" : format_currency(Math.abs(value), currency))
|
||||||
|
+ " " + dr_or_cr
|
||||||
|
+ "</div>";
|
||||||
|
};
|
||||||
|
this.frm.fields_dict.accounts.grid.update_docfield_property('balance', 'formatter', formatter);
|
||||||
|
this.frm.fields_dict.accounts.grid.update_docfield_property('party_balance', 'formatter', formatter);
|
||||||
}
|
}
|
||||||
|
|
||||||
reference_name(doc, cdt, cdn) {
|
reference_name(doc, cdt, cdn) {
|
||||||
var d = frappe.get_doc(cdt, cdn);
|
var d = frappe.get_doc(cdt, cdn);
|
||||||
|
|
||||||
if (d.reference_name) {
|
if(d.reference_name) {
|
||||||
if (d.reference_type === "Purchase Invoice" && !flt(d.debit)) {
|
if (d.reference_type==="Purchase Invoice" && !flt(d.debit)) {
|
||||||
this.get_outstanding("Purchase Invoice", d.reference_name, doc.company, d);
|
this.get_outstanding('Purchase Invoice', d.reference_name, doc.company, d);
|
||||||
} else if (d.reference_type === "Sales Invoice" && !flt(d.credit)) {
|
} else if (d.reference_type==="Sales Invoice" && !flt(d.credit)) {
|
||||||
this.get_outstanding("Sales Invoice", d.reference_name, doc.company, d);
|
this.get_outstanding('Sales Invoice', d.reference_name, doc.company, d);
|
||||||
} else if (d.reference_type === "Journal Entry" && !flt(d.credit) && !flt(d.debit)) {
|
} else if (d.reference_type==="Journal Entry" && !flt(d.credit) && !flt(d.debit)) {
|
||||||
this.get_outstanding("Journal Entry", d.reference_name, doc.company, d);
|
this.get_outstanding('Journal Entry', d.reference_name, doc.company, d);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
get_outstanding(doctype, docname, company, child) {
|
get_outstanding(doctype, docname, company, child) {
|
||||||
var args = {
|
var args = {
|
||||||
doctype: doctype,
|
"doctype": doctype,
|
||||||
docname: docname,
|
"docname": docname,
|
||||||
party: child.party,
|
"party": child.party,
|
||||||
account: child.account,
|
"account": child.account,
|
||||||
account_currency: child.account_currency,
|
"account_currency": child.account_currency,
|
||||||
company: company,
|
"company": company
|
||||||
};
|
}
|
||||||
|
|
||||||
return frappe.call({
|
return frappe.call({
|
||||||
method: "erpnext.accounts.doctype.journal_entry.journal_entry.get_outstanding",
|
method: "erpnext.accounts.doctype.journal_entry.journal_entry.get_outstanding",
|
||||||
args: { args: args },
|
args: { args: args},
|
||||||
callback: function (r) {
|
callback: function(r) {
|
||||||
if (r.message) {
|
if(r.message) {
|
||||||
$.each(r.message, function (field, value) {
|
$.each(r.message, function(field, value) {
|
||||||
frappe.model.set_value(child.doctype, child.name, field, value);
|
frappe.model.set_value(child.doctype, child.name, field, value);
|
||||||
});
|
})
|
||||||
}
|
}
|
||||||
},
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
accounts_add(doc, cdt, cdn) {
|
accounts_add(doc, cdt, cdn) {
|
||||||
var row = frappe.get_doc(cdt, cdn);
|
var row = frappe.get_doc(cdt, cdn);
|
||||||
$.each(doc.accounts, function (i, d) {
|
$.each(doc.accounts, function(i, d) {
|
||||||
if (d.account && d.party && d.party_type) {
|
if(d.account && d.party && d.party_type) {
|
||||||
row.account = d.account;
|
row.account = d.account;
|
||||||
row.party = d.party;
|
row.party = d.party;
|
||||||
row.party_type = d.party_type;
|
row.party_type = d.party_type;
|
||||||
@@ -393,8 +363,8 @@ erpnext.accounts.JournalEntry = class JournalEntry extends frappe.ui.form.Contro
|
|||||||
});
|
});
|
||||||
|
|
||||||
// set difference
|
// set difference
|
||||||
if (doc.difference) {
|
if(doc.difference) {
|
||||||
if (doc.difference > 0) {
|
if(doc.difference > 0) {
|
||||||
row.credit_in_account_currency = doc.difference;
|
row.credit_in_account_currency = doc.difference;
|
||||||
row.credit = doc.difference;
|
row.credit = doc.difference;
|
||||||
} else {
|
} else {
|
||||||
@@ -404,141 +374,132 @@ erpnext.accounts.JournalEntry = class JournalEntry extends frappe.ui.form.Contro
|
|||||||
}
|
}
|
||||||
cur_frm.cscript.update_totals(doc);
|
cur_frm.cscript.update_totals(doc);
|
||||||
|
|
||||||
erpnext.accounts.dimensions.copy_dimension_from_first_row(this.frm, cdt, cdn, "accounts");
|
erpnext.accounts.dimensions.copy_dimension_from_first_row(this.frm, cdt, cdn, 'accounts');
|
||||||
}
|
}
|
||||||
|
|
||||||
};
|
};
|
||||||
|
|
||||||
cur_frm.script_manager.make(erpnext.accounts.JournalEntry);
|
cur_frm.script_manager.make(erpnext.accounts.JournalEntry);
|
||||||
|
|
||||||
cur_frm.cscript.update_totals = function (doc) {
|
cur_frm.cscript.update_totals = function(doc) {
|
||||||
var td = 0.0;
|
var td=0.0; var tc =0.0;
|
||||||
var tc = 0.0;
|
|
||||||
var accounts = doc.accounts || [];
|
var accounts = doc.accounts || [];
|
||||||
for (var i in accounts) {
|
for(var i in accounts) {
|
||||||
td += flt(accounts[i].debit, precision("debit", accounts[i]));
|
td += flt(accounts[i].debit, precision("debit", accounts[i]));
|
||||||
tc += flt(accounts[i].credit, precision("credit", accounts[i]));
|
tc += flt(accounts[i].credit, precision("credit", accounts[i]));
|
||||||
}
|
}
|
||||||
doc = locals[doc.doctype][doc.name];
|
doc = locals[doc.doctype][doc.name];
|
||||||
doc.total_debit = td;
|
doc.total_debit = td;
|
||||||
doc.total_credit = tc;
|
doc.total_credit = tc;
|
||||||
doc.difference = flt(td - tc, precision("difference"));
|
doc.difference = flt((td - tc), precision("difference"));
|
||||||
refresh_many(["total_debit", "total_credit", "difference"]);
|
refresh_many(['total_debit','total_credit','difference']);
|
||||||
};
|
}
|
||||||
|
|
||||||
cur_frm.cscript.get_balance = function (doc, dt, dn) {
|
cur_frm.cscript.get_balance = function(doc,dt,dn) {
|
||||||
cur_frm.cscript.update_totals(doc);
|
cur_frm.cscript.update_totals(doc);
|
||||||
cur_frm.call("get_balance", null, () => {
|
cur_frm.call('get_balance', null, () => { cur_frm.refresh(); });
|
||||||
cur_frm.refresh();
|
}
|
||||||
});
|
|
||||||
};
|
|
||||||
|
|
||||||
cur_frm.cscript.validate = function (doc, cdt, cdn) {
|
cur_frm.cscript.validate = function(doc,cdt,cdn) {
|
||||||
cur_frm.cscript.update_totals(doc);
|
cur_frm.cscript.update_totals(doc);
|
||||||
};
|
}
|
||||||
|
|
||||||
frappe.ui.form.on("Journal Entry Account", {
|
frappe.ui.form.on("Journal Entry Account", {
|
||||||
party: function (frm, cdt, cdn) {
|
party: function(frm, cdt, cdn) {
|
||||||
var d = frappe.get_doc(cdt, cdn);
|
var d = frappe.get_doc(cdt, cdn);
|
||||||
if (!d.account && d.party_type && d.party) {
|
if(!d.account && d.party_type && d.party) {
|
||||||
if (!frm.doc.company) frappe.throw(__("Please select Company"));
|
if(!frm.doc.company) frappe.throw(__("Please select Company"));
|
||||||
return frm.call({
|
return frm.call({
|
||||||
method: "erpnext.accounts.doctype.journal_entry.journal_entry.get_party_account_and_currency",
|
method: "erpnext.accounts.doctype.journal_entry.journal_entry.get_party_account_and_balance",
|
||||||
child: d,
|
child: d,
|
||||||
args: {
|
args: {
|
||||||
company: frm.doc.company,
|
company: frm.doc.company,
|
||||||
party_type: d.party_type,
|
party_type: d.party_type,
|
||||||
party: d.party,
|
party: d.party,
|
||||||
},
|
cost_center: d.cost_center
|
||||||
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
cost_center: function (frm, dt, dn) {
|
cost_center: function(frm, dt, dn) {
|
||||||
erpnext.journal_entry.set_account_details(frm, dt, dn);
|
erpnext.journal_entry.set_account_balance(frm, dt, dn);
|
||||||
},
|
},
|
||||||
|
|
||||||
account: function (frm, dt, dn) {
|
account: function(frm, dt, dn) {
|
||||||
erpnext.journal_entry.set_account_details(frm, dt, dn);
|
erpnext.journal_entry.set_account_balance(frm, dt, dn);
|
||||||
},
|
},
|
||||||
|
|
||||||
debit_in_account_currency: function (frm, cdt, cdn) {
|
debit_in_account_currency: function(frm, cdt, cdn) {
|
||||||
erpnext.journal_entry.set_exchange_rate(frm, cdt, cdn);
|
erpnext.journal_entry.set_exchange_rate(frm, cdt, cdn);
|
||||||
},
|
},
|
||||||
|
|
||||||
credit_in_account_currency: function (frm, cdt, cdn) {
|
credit_in_account_currency: function(frm, cdt, cdn) {
|
||||||
erpnext.journal_entry.set_exchange_rate(frm, cdt, cdn);
|
erpnext.journal_entry.set_exchange_rate(frm, cdt, cdn);
|
||||||
},
|
},
|
||||||
|
|
||||||
debit: function (frm, dt, dn) {
|
debit: function(frm, dt, dn) {
|
||||||
cur_frm.cscript.update_totals(frm.doc);
|
cur_frm.cscript.update_totals(frm.doc);
|
||||||
},
|
},
|
||||||
|
|
||||||
credit: function (frm, dt, dn) {
|
credit: function(frm, dt, dn) {
|
||||||
cur_frm.cscript.update_totals(frm.doc);
|
cur_frm.cscript.update_totals(frm.doc);
|
||||||
},
|
},
|
||||||
|
|
||||||
exchange_rate: function (frm, cdt, cdn) {
|
exchange_rate: function(frm, cdt, cdn) {
|
||||||
var company_currency = frappe.get_doc(":Company", frm.doc.company).default_currency;
|
var company_currency = frappe.get_doc(":Company", frm.doc.company).default_currency;
|
||||||
var row = locals[cdt][cdn];
|
var row = locals[cdt][cdn];
|
||||||
|
|
||||||
if (row.account_currency == company_currency || !frm.doc.multi_currency) {
|
if(row.account_currency == company_currency || !frm.doc.multi_currency) {
|
||||||
frappe.model.set_value(cdt, cdn, "exchange_rate", 1);
|
frappe.model.set_value(cdt, cdn, "exchange_rate", 1);
|
||||||
}
|
}
|
||||||
|
|
||||||
erpnext.journal_entry.set_debit_credit_in_company_currency(frm, cdt, cdn);
|
erpnext.journal_entry.set_debit_credit_in_company_currency(frm, cdt, cdn);
|
||||||
},
|
}
|
||||||
});
|
})
|
||||||
|
|
||||||
frappe.ui.form.on("Journal Entry Account", "accounts_remove", function (frm) {
|
frappe.ui.form.on("Journal Entry Account", "accounts_remove", function(frm) {
|
||||||
cur_frm.cscript.update_totals(frm.doc);
|
cur_frm.cscript.update_totals(frm.doc);
|
||||||
});
|
});
|
||||||
|
|
||||||
$.extend(erpnext.journal_entry, {
|
$.extend(erpnext.journal_entry, {
|
||||||
toggle_fields_based_on_currency: function (frm) {
|
toggle_fields_based_on_currency: function(frm) {
|
||||||
var fields = ["currency_section", "account_currency", "exchange_rate", "debit", "credit"];
|
var fields = ["currency_section", "account_currency", "exchange_rate", "debit", "credit"];
|
||||||
|
|
||||||
var grid = frm.get_field("accounts").grid;
|
var grid = frm.get_field("accounts").grid;
|
||||||
if (grid) grid.set_column_disp(fields, frm.doc.multi_currency);
|
if(grid) grid.set_column_disp(fields, frm.doc.multi_currency);
|
||||||
|
|
||||||
// dynamic label
|
// dynamic label
|
||||||
var field_label_map = {
|
var field_label_map = {
|
||||||
debit_in_account_currency: "Debit",
|
"debit_in_account_currency": "Debit",
|
||||||
credit_in_account_currency: "Credit",
|
"credit_in_account_currency": "Credit"
|
||||||
};
|
};
|
||||||
|
|
||||||
$.each(field_label_map, function (fieldname, label) {
|
$.each(field_label_map, function (fieldname, label) {
|
||||||
frm.fields_dict.accounts.grid.update_docfield_property(
|
frm.fields_dict.accounts.grid.update_docfield_property(
|
||||||
fieldname,
|
fieldname,
|
||||||
"label",
|
'label',
|
||||||
frm.doc.multi_currency ? label + " in Account Currency" : label
|
frm.doc.multi_currency ? (label + " in Account Currency") : label
|
||||||
);
|
);
|
||||||
});
|
})
|
||||||
},
|
},
|
||||||
|
|
||||||
set_debit_credit_in_company_currency: function (frm, cdt, cdn) {
|
set_debit_credit_in_company_currency: function(frm, cdt, cdn) {
|
||||||
var row = locals[cdt][cdn];
|
var row = locals[cdt][cdn];
|
||||||
|
|
||||||
frappe.model.set_value(
|
frappe.model.set_value(cdt, cdn, "debit",
|
||||||
cdt,
|
flt(flt(row.debit_in_account_currency)*row.exchange_rate, precision("debit", row)));
|
||||||
cdn,
|
|
||||||
"debit",
|
|
||||||
flt(flt(row.debit_in_account_currency) * row.exchange_rate, precision("debit", row))
|
|
||||||
);
|
|
||||||
|
|
||||||
frappe.model.set_value(
|
frappe.model.set_value(cdt, cdn, "credit",
|
||||||
cdt,
|
flt(flt(row.credit_in_account_currency)*row.exchange_rate, precision("credit", row)));
|
||||||
cdn,
|
|
||||||
"credit",
|
|
||||||
flt(flt(row.credit_in_account_currency) * row.exchange_rate, precision("credit", row))
|
|
||||||
);
|
|
||||||
|
|
||||||
cur_frm.cscript.update_totals(frm.doc);
|
cur_frm.cscript.update_totals(frm.doc);
|
||||||
},
|
},
|
||||||
|
|
||||||
set_exchange_rate: function (frm, cdt, cdn) {
|
set_exchange_rate: function(frm, cdt, cdn) {
|
||||||
var company_currency = frappe.get_doc(":Company", frm.doc.company).default_currency;
|
var company_currency = frappe.get_doc(":Company", frm.doc.company).default_currency;
|
||||||
var row = locals[cdt][cdn];
|
var row = locals[cdt][cdn];
|
||||||
|
|
||||||
if (row.account_currency == company_currency || !frm.doc.multi_currency) {
|
if(row.account_currency == company_currency || !frm.doc.multi_currency) {
|
||||||
row.exchange_rate = 1;
|
row.exchange_rate = 1;
|
||||||
erpnext.journal_entry.set_debit_credit_in_company_currency(frm, cdt, cdn);
|
erpnext.journal_entry.set_debit_credit_in_company_currency(frm, cdt, cdn);
|
||||||
} else if (!row.exchange_rate || row.exchange_rate == 1 || row.account_type == "Bank") {
|
} else if (!row.exchange_rate || row.exchange_rate == 1 || row.account_type == "Bank") {
|
||||||
@@ -553,70 +514,50 @@ $.extend(erpnext.journal_entry, {
|
|||||||
reference_name: cstr(row.reference_name),
|
reference_name: cstr(row.reference_name),
|
||||||
debit: flt(row.debit_in_account_currency),
|
debit: flt(row.debit_in_account_currency),
|
||||||
credit: flt(row.credit_in_account_currency),
|
credit: flt(row.credit_in_account_currency),
|
||||||
exchange_rate: row.exchange_rate,
|
exchange_rate: row.exchange_rate
|
||||||
},
|
},
|
||||||
callback: function (r) {
|
callback: function(r) {
|
||||||
if (r.message) {
|
if(r.message) {
|
||||||
row.exchange_rate = r.message;
|
row.exchange_rate = r.message;
|
||||||
erpnext.journal_entry.set_debit_credit_in_company_currency(frm, cdt, cdn);
|
erpnext.journal_entry.set_debit_credit_in_company_currency(frm, cdt, cdn);
|
||||||
}
|
}
|
||||||
},
|
}
|
||||||
});
|
})
|
||||||
} else {
|
} else {
|
||||||
erpnext.journal_entry.set_debit_credit_in_company_currency(frm, cdt, cdn);
|
erpnext.journal_entry.set_debit_credit_in_company_currency(frm, cdt, cdn);
|
||||||
}
|
}
|
||||||
refresh_field("exchange_rate", cdn, "accounts");
|
refresh_field("exchange_rate", cdn, "accounts");
|
||||||
},
|
},
|
||||||
|
|
||||||
quick_entry: function (frm) {
|
quick_entry: function(frm) {
|
||||||
var naming_series_options = frm.fields_dict.naming_series.df.options;
|
var naming_series_options = frm.fields_dict.naming_series.df.options;
|
||||||
var naming_series_default =
|
var naming_series_default = frm.fields_dict.naming_series.df.default || naming_series_options.split("\n")[0];
|
||||||
frm.fields_dict.naming_series.df.default || naming_series_options.split("\n")[0];
|
|
||||||
|
|
||||||
var dialog = new frappe.ui.Dialog({
|
var dialog = new frappe.ui.Dialog({
|
||||||
title: __("Quick Journal Entry"),
|
title: __("Quick Journal Entry"),
|
||||||
fields: [
|
fields: [
|
||||||
{ fieldtype: "Currency", fieldname: "debit", label: __("Amount"), reqd: 1 },
|
{fieldtype: "Currency", fieldname: "debit", label: __("Amount"), reqd: 1},
|
||||||
{
|
{fieldtype: "Link", fieldname: "debit_account", label: __("Debit Account"), reqd: 1,
|
||||||
fieldtype: "Link",
|
|
||||||
fieldname: "debit_account",
|
|
||||||
label: __("Debit Account"),
|
|
||||||
reqd: 1,
|
|
||||||
options: "Account",
|
options: "Account",
|
||||||
get_query: function () {
|
get_query: function() {
|
||||||
return erpnext.journal_entry.account_query(frm);
|
return erpnext.journal_entry.account_query(frm);
|
||||||
},
|
}
|
||||||
},
|
},
|
||||||
{
|
{fieldtype: "Link", fieldname: "credit_account", label: __("Credit Account"), reqd: 1,
|
||||||
fieldtype: "Link",
|
|
||||||
fieldname: "credit_account",
|
|
||||||
label: __("Credit Account"),
|
|
||||||
reqd: 1,
|
|
||||||
options: "Account",
|
options: "Account",
|
||||||
get_query: function () {
|
get_query: function() {
|
||||||
return erpnext.journal_entry.account_query(frm);
|
return erpnext.journal_entry.account_query(frm);
|
||||||
},
|
}
|
||||||
},
|
},
|
||||||
{
|
{fieldtype: "Date", fieldname: "posting_date", label: __("Date"), reqd: 1,
|
||||||
fieldtype: "Date",
|
default: frm.doc.posting_date},
|
||||||
fieldname: "posting_date",
|
{fieldtype: "Small Text", fieldname: "user_remark", label: __("User Remark")},
|
||||||
label: __("Date"),
|
{fieldtype: "Select", fieldname: "naming_series", label: __("Series"), reqd: 1,
|
||||||
reqd: 1,
|
options: naming_series_options, default: naming_series_default},
|
||||||
default: frm.doc.posting_date,
|
]
|
||||||
},
|
|
||||||
{ fieldtype: "Small Text", fieldname: "user_remark", label: __("User Remark") },
|
|
||||||
{
|
|
||||||
fieldtype: "Select",
|
|
||||||
fieldname: "naming_series",
|
|
||||||
label: __("Series"),
|
|
||||||
reqd: 1,
|
|
||||||
options: naming_series_options,
|
|
||||||
default: naming_series_default,
|
|
||||||
},
|
|
||||||
],
|
|
||||||
});
|
});
|
||||||
|
|
||||||
dialog.set_primary_action(__("Save"), function () {
|
dialog.set_primary_action(__("Save"), function() {
|
||||||
var btn = this;
|
var btn = this;
|
||||||
var values = dialog.get_values();
|
var values = dialog.get_values();
|
||||||
|
|
||||||
@@ -633,21 +574,11 @@ $.extend(erpnext.journal_entry, {
|
|||||||
|
|
||||||
var debit_row = frm.fields_dict.accounts.grid.add_new_row();
|
var debit_row = frm.fields_dict.accounts.grid.add_new_row();
|
||||||
frappe.model.set_value(debit_row.doctype, debit_row.name, "account", values.debit_account);
|
frappe.model.set_value(debit_row.doctype, debit_row.name, "account", values.debit_account);
|
||||||
frappe.model.set_value(
|
frappe.model.set_value(debit_row.doctype, debit_row.name, "debit_in_account_currency", values.debit);
|
||||||
debit_row.doctype,
|
|
||||||
debit_row.name,
|
|
||||||
"debit_in_account_currency",
|
|
||||||
values.debit
|
|
||||||
);
|
|
||||||
|
|
||||||
var credit_row = frm.fields_dict.accounts.grid.add_new_row();
|
var credit_row = frm.fields_dict.accounts.grid.add_new_row();
|
||||||
frappe.model.set_value(credit_row.doctype, credit_row.name, "account", values.credit_account);
|
frappe.model.set_value(credit_row.doctype, credit_row.name, "account", values.credit_account);
|
||||||
frappe.model.set_value(
|
frappe.model.set_value(credit_row.doctype, credit_row.name, "credit_in_account_currency", values.debit);
|
||||||
credit_row.doctype,
|
|
||||||
credit_row.name,
|
|
||||||
"credit_in_account_currency",
|
|
||||||
values.debit
|
|
||||||
);
|
|
||||||
|
|
||||||
frm.save();
|
frm.save();
|
||||||
|
|
||||||
@@ -657,39 +588,51 @@ $.extend(erpnext.journal_entry, {
|
|||||||
dialog.show();
|
dialog.show();
|
||||||
},
|
},
|
||||||
|
|
||||||
account_query: function (frm) {
|
account_query: function(frm) {
|
||||||
var filters = {
|
var filters = {
|
||||||
company: frm.doc.company,
|
company: frm.doc.company,
|
||||||
is_group: 0,
|
is_group: 0
|
||||||
};
|
};
|
||||||
if (!frm.doc.multi_currency) {
|
if(!frm.doc.multi_currency) {
|
||||||
$.extend(filters, {
|
$.extend(filters, {
|
||||||
account_currency: [
|
account_currency: ['in', [frappe.get_doc(":Company", frm.doc.company).default_currency, null]]
|
||||||
"in",
|
|
||||||
[frappe.get_doc(":Company", frm.doc.company).default_currency, null],
|
|
||||||
],
|
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
return { filters: filters };
|
return { filters: filters };
|
||||||
},
|
},
|
||||||
|
|
||||||
reverse_journal_entry: function () {
|
against_account_query: function(frm) {
|
||||||
|
if (frm.doc.against_type != "Account"){
|
||||||
|
return { filters: {} };
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
let filters = { company: frm.doc.company, is_group: 0 };
|
||||||
|
if(!frm.doc.multi_currency) {
|
||||||
|
$.extend(filters, {
|
||||||
|
account_currency: ['in', [frappe.get_doc(":Company", frm.doc.company).default_currency, null]]
|
||||||
|
});
|
||||||
|
}
|
||||||
|
return { filters: filters };
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
reverse_journal_entry: function() {
|
||||||
frappe.model.open_mapped_doc({
|
frappe.model.open_mapped_doc({
|
||||||
method: "erpnext.accounts.doctype.journal_entry.journal_entry.make_reverse_journal_entry",
|
method: "erpnext.accounts.doctype.journal_entry.journal_entry.make_reverse_journal_entry",
|
||||||
frm: cur_frm,
|
frm: cur_frm
|
||||||
});
|
})
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
$.extend(erpnext.journal_entry, {
|
$.extend(erpnext.journal_entry, {
|
||||||
set_account_details: function (frm, dt, dn) {
|
set_account_balance: function(frm, dt, dn) {
|
||||||
var d = locals[dt][dn];
|
var d = locals[dt][dn];
|
||||||
if (d.account) {
|
if(d.account) {
|
||||||
if (!frm.doc.company) frappe.throw(__("Please select Company first"));
|
if(!frm.doc.company) frappe.throw(__("Please select Company first"));
|
||||||
if (!frm.doc.posting_date) frappe.throw(__("Please select Posting Date first"));
|
if(!frm.doc.posting_date) frappe.throw(__("Please select Posting Date first"));
|
||||||
|
|
||||||
return frappe.call({
|
return frappe.call({
|
||||||
method: "erpnext.accounts.doctype.journal_entry.journal_entry.get_account_details_and_party_type",
|
method: "erpnext.accounts.doctype.journal_entry.journal_entry.get_account_balance_and_party_type",
|
||||||
args: {
|
args: {
|
||||||
account: d.account,
|
account: d.account,
|
||||||
date: frm.doc.posting_date,
|
date: frm.doc.posting_date,
|
||||||
@@ -697,14 +640,15 @@ $.extend(erpnext.journal_entry, {
|
|||||||
debit: flt(d.debit_in_account_currency),
|
debit: flt(d.debit_in_account_currency),
|
||||||
credit: flt(d.credit_in_account_currency),
|
credit: flt(d.credit_in_account_currency),
|
||||||
exchange_rate: d.exchange_rate,
|
exchange_rate: d.exchange_rate,
|
||||||
|
cost_center: d.cost_center
|
||||||
},
|
},
|
||||||
callback: function (r) {
|
callback: function(r) {
|
||||||
if (r.message) {
|
if(r.message) {
|
||||||
$.extend(d, r.message);
|
$.extend(d, r.message);
|
||||||
erpnext.journal_entry.set_debit_credit_in_company_currency(frm, dt, dn);
|
erpnext.journal_entry.set_debit_credit_in_company_currency(frm, dt, dn);
|
||||||
refresh_field("accounts");
|
refresh_field('accounts');
|
||||||
}
|
}
|
||||||
},
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -64,8 +64,7 @@
|
|||||||
"stock_entry",
|
"stock_entry",
|
||||||
"subscription_section",
|
"subscription_section",
|
||||||
"auto_repeat",
|
"auto_repeat",
|
||||||
"amended_from",
|
"amended_from"
|
||||||
"repost_required"
|
|
||||||
],
|
],
|
||||||
"fields": [
|
"fields": [
|
||||||
{
|
{
|
||||||
@@ -544,15 +543,6 @@
|
|||||||
"label": "Is System Generated",
|
"label": "Is System Generated",
|
||||||
"no_copy": 1,
|
"no_copy": 1,
|
||||||
"read_only": 1
|
"read_only": 1
|
||||||
},
|
|
||||||
{
|
|
||||||
"default": "0",
|
|
||||||
"fieldname": "repost_required",
|
|
||||||
"fieldtype": "Check",
|
|
||||||
"hidden": 1,
|
|
||||||
"label": "Repost Required",
|
|
||||||
"print_hide": 1,
|
|
||||||
"read_only": 1
|
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"icon": "fa fa-file-text",
|
"icon": "fa fa-file-text",
|
||||||
|
|||||||
@@ -13,10 +13,6 @@ from erpnext.accounts.deferred_revenue import get_deferred_booking_accounts
|
|||||||
from erpnext.accounts.doctype.invoice_discounting.invoice_discounting import (
|
from erpnext.accounts.doctype.invoice_discounting.invoice_discounting import (
|
||||||
get_party_account_based_on_invoice_discounting,
|
get_party_account_based_on_invoice_discounting,
|
||||||
)
|
)
|
||||||
from erpnext.accounts.doctype.repost_accounting_ledger.repost_accounting_ledger import (
|
|
||||||
validate_docs_for_deferred_accounting,
|
|
||||||
validate_docs_for_voucher_types,
|
|
||||||
)
|
|
||||||
from erpnext.accounts.doctype.tax_withholding_category.tax_withholding_category import (
|
from erpnext.accounts.doctype.tax_withholding_category.tax_withholding_category import (
|
||||||
get_party_tax_withholding_details,
|
get_party_tax_withholding_details,
|
||||||
)
|
)
|
||||||
@@ -144,6 +140,7 @@ class JournalEntry(AccountsController):
|
|||||||
self.set_print_format_fields()
|
self.set_print_format_fields()
|
||||||
self.validate_credit_debit_note()
|
self.validate_credit_debit_note()
|
||||||
self.validate_empty_accounts_table()
|
self.validate_empty_accounts_table()
|
||||||
|
self.set_account_and_party_balance()
|
||||||
self.validate_inter_company_accounts()
|
self.validate_inter_company_accounts()
|
||||||
self.validate_depr_entry_voucher_type()
|
self.validate_depr_entry_voucher_type()
|
||||||
|
|
||||||
@@ -153,24 +150,6 @@ class JournalEntry(AccountsController):
|
|||||||
if not self.title:
|
if not self.title:
|
||||||
self.title = self.get_title()
|
self.title = self.get_title()
|
||||||
|
|
||||||
def validate_for_repost(self):
|
|
||||||
validate_docs_for_voucher_types(["Journal Entry"])
|
|
||||||
validate_docs_for_deferred_accounting([self.name], [])
|
|
||||||
|
|
||||||
def submit(self):
|
|
||||||
if len(self.accounts) > 100:
|
|
||||||
msgprint(_("The task has been enqueued as a background job."), alert=True)
|
|
||||||
self.queue_action("submit", timeout=4600)
|
|
||||||
else:
|
|
||||||
return self._submit()
|
|
||||||
|
|
||||||
def cancel(self):
|
|
||||||
if len(self.accounts) > 100:
|
|
||||||
msgprint(_("The task has been enqueued as a background job."), alert=True)
|
|
||||||
self.queue_action("cancel", timeout=4600)
|
|
||||||
else:
|
|
||||||
return self._cancel()
|
|
||||||
|
|
||||||
def on_submit(self):
|
def on_submit(self):
|
||||||
self.validate_cheque_info()
|
self.validate_cheque_info()
|
||||||
self.check_credit_limit()
|
self.check_credit_limit()
|
||||||
@@ -180,15 +159,6 @@ class JournalEntry(AccountsController):
|
|||||||
self.update_inter_company_jv()
|
self.update_inter_company_jv()
|
||||||
self.update_invoice_discounting()
|
self.update_invoice_discounting()
|
||||||
|
|
||||||
def on_update_after_submit(self):
|
|
||||||
if hasattr(self, "repost_required"):
|
|
||||||
self.needs_repost = self.check_if_fields_updated(
|
|
||||||
fields_to_check=[], child_tables={"accounts": []}
|
|
||||||
)
|
|
||||||
if self.needs_repost:
|
|
||||||
self.validate_for_repost()
|
|
||||||
self.db_set("repost_required", self.needs_repost)
|
|
||||||
|
|
||||||
def on_cancel(self):
|
def on_cancel(self):
|
||||||
# References for this Journal are removed on the `on_cancel` event in accounts_controller
|
# References for this Journal are removed on the `on_cancel` event in accounts_controller
|
||||||
super(JournalEntry, self).on_cancel()
|
super(JournalEntry, self).on_cancel()
|
||||||
@@ -334,6 +304,7 @@ class JournalEntry(AccountsController):
|
|||||||
"account": tax_withholding_details.get("account_head"),
|
"account": tax_withholding_details.get("account_head"),
|
||||||
rev_debit_or_credit: tax_withholding_details.get("tax_amount"),
|
rev_debit_or_credit: tax_withholding_details.get("tax_amount"),
|
||||||
"against_account": parties[0],
|
"against_account": parties[0],
|
||||||
|
"against_account_link": parties[0],
|
||||||
},
|
},
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -575,28 +546,17 @@ class JournalEntry(AccountsController):
|
|||||||
elif d.party_type == "Supplier" and flt(d.credit) > 0:
|
elif d.party_type == "Supplier" and flt(d.credit) > 0:
|
||||||
frappe.throw(_("Row {0}: Advance against Supplier must be debit").format(d.idx))
|
frappe.throw(_("Row {0}: Advance against Supplier must be debit").format(d.idx))
|
||||||
|
|
||||||
def system_generated_gain_loss(self):
|
|
||||||
return (
|
|
||||||
self.voucher_type == "Exchange Gain Or Loss"
|
|
||||||
and self.multi_currency
|
|
||||||
and self.is_system_generated
|
|
||||||
)
|
|
||||||
|
|
||||||
def validate_against_jv(self):
|
def validate_against_jv(self):
|
||||||
for d in self.get("accounts"):
|
for d in self.get("accounts"):
|
||||||
if d.reference_type == "Journal Entry":
|
if d.reference_type == "Journal Entry":
|
||||||
account_root_type = frappe.get_cached_value("Account", d.account, "root_type")
|
account_root_type = frappe.get_cached_value("Account", d.account, "root_type")
|
||||||
if account_root_type == "Asset" and flt(d.debit) > 0 and not self.system_generated_gain_loss():
|
if account_root_type == "Asset" and flt(d.debit) > 0:
|
||||||
frappe.throw(
|
frappe.throw(
|
||||||
_(
|
_(
|
||||||
"Row #{0}: For {1}, you can select reference document only if account gets credited"
|
"Row #{0}: For {1}, you can select reference document only if account gets credited"
|
||||||
).format(d.idx, d.account)
|
).format(d.idx, d.account)
|
||||||
)
|
)
|
||||||
elif (
|
elif account_root_type == "Liability" and flt(d.credit) > 0:
|
||||||
account_root_type == "Liability"
|
|
||||||
and flt(d.credit) > 0
|
|
||||||
and not self.system_generated_gain_loss()
|
|
||||||
):
|
|
||||||
frappe.throw(
|
frappe.throw(
|
||||||
_(
|
_(
|
||||||
"Row #{0}: For {1}, you can select reference document only if account gets debited"
|
"Row #{0}: For {1}, you can select reference document only if account gets debited"
|
||||||
@@ -628,7 +588,7 @@ class JournalEntry(AccountsController):
|
|||||||
for jvd in against_entries:
|
for jvd in against_entries:
|
||||||
if flt(jvd[dr_or_cr]) > 0:
|
if flt(jvd[dr_or_cr]) > 0:
|
||||||
valid = True
|
valid = True
|
||||||
if not valid and not self.system_generated_gain_loss():
|
if not valid:
|
||||||
frappe.throw(
|
frappe.throw(
|
||||||
_("Against Journal Entry {0} does not have any unmatched {1} entry").format(
|
_("Against Journal Entry {0} does not have any unmatched {1} entry").format(
|
||||||
d.reference_name, dr_or_cr
|
d.reference_name, dr_or_cr
|
||||||
@@ -672,7 +632,7 @@ class JournalEntry(AccountsController):
|
|||||||
)
|
)
|
||||||
|
|
||||||
# set totals
|
# set totals
|
||||||
if not d.reference_name in self.reference_totals:
|
if d.reference_name not in self.reference_totals:
|
||||||
self.reference_totals[d.reference_name] = 0.0
|
self.reference_totals[d.reference_name] = 0.0
|
||||||
|
|
||||||
if self.voucher_type not in ("Deferred Revenue", "Deferred Expense"):
|
if self.voucher_type not in ("Deferred Revenue", "Deferred Expense"):
|
||||||
@@ -791,27 +751,90 @@ class JournalEntry(AccountsController):
|
|||||||
)
|
)
|
||||||
|
|
||||||
def set_against_account(self):
|
def set_against_account(self):
|
||||||
accounts_debited, accounts_credited = [], []
|
|
||||||
if self.voucher_type in ("Deferred Revenue", "Deferred Expense"):
|
if self.voucher_type in ("Deferred Revenue", "Deferred Expense"):
|
||||||
for d in self.get("accounts"):
|
for d in self.get("accounts"):
|
||||||
if d.reference_type == "Sales Invoice":
|
if d.reference_type == "Sales Invoice":
|
||||||
field = "customer"
|
against_type = "Customer"
|
||||||
else:
|
else:
|
||||||
field = "supplier"
|
against_type = "Supplier"
|
||||||
|
|
||||||
d.against_account = frappe.db.get_value(d.reference_type, d.reference_name, field)
|
against_account = frappe.db.get_value(d.reference_type, d.reference_name, against_type.lower())
|
||||||
|
d.against_type = against_type
|
||||||
|
d.against_account_link = against_account
|
||||||
else:
|
else:
|
||||||
for d in self.get("accounts"):
|
self.get_debited_credited_accounts()
|
||||||
if flt(d.debit) > 0:
|
if len(self.accounts_credited) > 1 and len(self.accounts_debited) > 1:
|
||||||
accounts_debited.append(d.party or d.account)
|
self.auto_set_against_accounts()
|
||||||
if flt(d.credit) > 0:
|
return
|
||||||
accounts_credited.append(d.party or d.account)
|
self.get_against_accounts()
|
||||||
|
|
||||||
for d in self.get("accounts"):
|
def auto_set_against_accounts(self):
|
||||||
if flt(d.debit) > 0:
|
for i in range(0, len(self.accounts), 2):
|
||||||
d.against_account = ", ".join(list(set(accounts_credited)))
|
acc = self.accounts[i]
|
||||||
if flt(d.credit) > 0:
|
against_acc = self.accounts[i + 1]
|
||||||
d.against_account = ", ".join(list(set(accounts_debited)))
|
if acc.debit_in_account_currency > 0:
|
||||||
|
current_val = acc.debit_in_account_currency * flt(acc.exchange_rate)
|
||||||
|
against_val = against_acc.credit_in_account_currency * flt(against_acc.exchange_rate)
|
||||||
|
else:
|
||||||
|
current_val = acc.credit_in_account_currency * flt(acc.exchange_rate)
|
||||||
|
against_val = against_acc.debit_in_account_currency * flt(against_acc.exchange_rate)
|
||||||
|
|
||||||
|
if current_val == against_val:
|
||||||
|
acc.against_type = against_acc.party_type or "Account"
|
||||||
|
against_acc.against_type = acc.party_type or "Account"
|
||||||
|
|
||||||
|
acc.against_account_link = against_acc.party or against_acc.account
|
||||||
|
against_acc.against_account_link = acc.party or acc.account
|
||||||
|
else:
|
||||||
|
frappe.msgprint(
|
||||||
|
_(
|
||||||
|
"Unable to automatically determine {0} accounts. Set them up in the {1} table if needed."
|
||||||
|
).format(frappe.bold("against"), frappe.bold("Accounting Entries")),
|
||||||
|
alert=True,
|
||||||
|
)
|
||||||
|
break
|
||||||
|
|
||||||
|
def get_against_accounts(self):
|
||||||
|
self.against_accounts = []
|
||||||
|
self.split_account = {}
|
||||||
|
self.get_debited_credited_accounts()
|
||||||
|
|
||||||
|
if self.separate_against_account_entries:
|
||||||
|
no_of_credited_acc, no_of_debited_acc = len(self.accounts_credited), len(self.accounts_debited)
|
||||||
|
if no_of_credited_acc <= 1 and no_of_debited_acc <= 1:
|
||||||
|
self.set_against_accounts_for_single_dr_cr()
|
||||||
|
self.separate_against_account_entries = 0
|
||||||
|
elif no_of_credited_acc == 1:
|
||||||
|
self.against_accounts = self.accounts_debited
|
||||||
|
self.split_account = self.accounts_credited[0]
|
||||||
|
elif no_of_debited_acc == 1:
|
||||||
|
self.against_accounts = self.accounts_credited
|
||||||
|
self.split_account = self.accounts_debited[0]
|
||||||
|
|
||||||
|
def get_debited_credited_accounts(self):
|
||||||
|
self.accounts_debited, self.accounts_credited = [], []
|
||||||
|
self.separate_against_account_entries = 1
|
||||||
|
for d in self.get("accounts"):
|
||||||
|
if flt(d.debit) > 0:
|
||||||
|
self.accounts_debited.append(d)
|
||||||
|
elif flt(d.credit) > 0:
|
||||||
|
self.accounts_credited.append(d)
|
||||||
|
|
||||||
|
if d.against_account_link:
|
||||||
|
self.separate_against_account_entries = 0
|
||||||
|
break
|
||||||
|
|
||||||
|
def set_against_accounts_for_single_dr_cr(self):
|
||||||
|
against_account = None
|
||||||
|
for d in self.accounts:
|
||||||
|
if flt(d.debit) > 0:
|
||||||
|
against_account = self.accounts_credited[0]
|
||||||
|
elif flt(d.credit) > 0:
|
||||||
|
against_account = self.accounts_debited[0]
|
||||||
|
if against_account:
|
||||||
|
d.against_type = against_account.party_type or "Account"
|
||||||
|
d.against_account = against_account.party or against_account.account
|
||||||
|
d.against_account_link = against_account.party or against_account.account
|
||||||
|
|
||||||
def validate_debit_credit_amount(self):
|
def validate_debit_credit_amount(self):
|
||||||
if not (self.voucher_type == "Exchange Gain Or Loss" and self.multi_currency):
|
if not (self.voucher_type == "Exchange Gain Or Loss" and self.multi_currency):
|
||||||
@@ -1008,42 +1031,108 @@ class JournalEntry(AccountsController):
|
|||||||
|
|
||||||
def build_gl_map(self):
|
def build_gl_map(self):
|
||||||
gl_map = []
|
gl_map = []
|
||||||
|
conversion_rate_map = self.get_conversion_rate_map()
|
||||||
|
transaction_currency_map = self.get_transaction_currency_map()
|
||||||
|
company_currency = erpnext.get_company_currency(self.company)
|
||||||
|
|
||||||
|
self.get_against_accounts()
|
||||||
for d in self.get("accounts"):
|
for d in self.get("accounts"):
|
||||||
if d.debit or d.credit or (self.voucher_type == "Exchange Gain Or Loss"):
|
if d.debit or d.credit or (self.voucher_type == "Exchange Gain Or Loss"):
|
||||||
r = [d.user_remark, self.remark]
|
r = [d.user_remark, self.remark]
|
||||||
r = [x for x in r if x]
|
r = [x for x in r if x]
|
||||||
remarks = "\n".join(r)
|
remarks = "\n".join(r)
|
||||||
|
|
||||||
gl_map.append(
|
gl_dict = self.get_gl_dict(
|
||||||
self.get_gl_dict(
|
{
|
||||||
{
|
"account": d.account,
|
||||||
"account": d.account,
|
"party_type": d.party_type,
|
||||||
"party_type": d.party_type,
|
"due_date": self.due_date,
|
||||||
"due_date": self.due_date,
|
"party": d.party,
|
||||||
"party": d.party,
|
"debit": flt(d.debit, d.precision("debit")),
|
||||||
"against": d.against_account,
|
"credit": flt(d.credit, d.precision("credit")),
|
||||||
"debit": flt(d.debit, d.precision("debit")),
|
"account_currency": d.account_currency,
|
||||||
"credit": flt(d.credit, d.precision("credit")),
|
"debit_in_account_currency": flt(
|
||||||
"account_currency": d.account_currency,
|
d.debit_in_account_currency, d.precision("debit_in_account_currency")
|
||||||
"debit_in_account_currency": flt(
|
),
|
||||||
d.debit_in_account_currency, d.precision("debit_in_account_currency")
|
"credit_in_account_currency": flt(
|
||||||
),
|
d.credit_in_account_currency, d.precision("credit_in_account_currency")
|
||||||
"credit_in_account_currency": flt(
|
),
|
||||||
d.credit_in_account_currency, d.precision("credit_in_account_currency")
|
"against_voucher_type": d.reference_type,
|
||||||
),
|
"against_voucher": d.reference_name,
|
||||||
"against_voucher_type": d.reference_type,
|
"remarks": remarks,
|
||||||
"against_voucher": d.reference_name,
|
"voucher_detail_no": d.reference_detail_no,
|
||||||
"remarks": remarks,
|
"cost_center": d.cost_center,
|
||||||
"voucher_detail_no": d.reference_detail_no,
|
"project": d.project,
|
||||||
"cost_center": d.cost_center,
|
"finance_book": self.finance_book,
|
||||||
"project": d.project,
|
"conversion_rate": conversion_rate_map.get(d.against_account_link, 1)
|
||||||
"finance_book": self.finance_book,
|
if d.account_currency == company_currency
|
||||||
},
|
else 1,
|
||||||
item=d,
|
"currency": transaction_currency_map.get(d.against_account_link, d.account_currency)
|
||||||
)
|
if d.account_currency == company_currency
|
||||||
|
else d.account_currency,
|
||||||
|
},
|
||||||
|
item=d,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
if not self.separate_against_account_entries:
|
||||||
|
gl_dict.update(
|
||||||
|
{
|
||||||
|
"against_type": d.against_type,
|
||||||
|
"against_link": d.against_account_link,
|
||||||
|
}
|
||||||
|
)
|
||||||
|
gl_map.append(gl_dict)
|
||||||
|
|
||||||
|
elif d in self.against_accounts:
|
||||||
|
gl_dict.update(
|
||||||
|
{
|
||||||
|
"against_type": self.split_account.get("party_type") or "Account",
|
||||||
|
"against": self.split_account.get("party") or self.split_account.get("account"),
|
||||||
|
"against_link": self.split_account.get("party") or self.split_account.get("account"),
|
||||||
|
}
|
||||||
|
)
|
||||||
|
gl_map.append(gl_dict)
|
||||||
|
|
||||||
|
else:
|
||||||
|
for against_account in self.against_accounts:
|
||||||
|
against_account = against_account.as_dict()
|
||||||
|
debit = against_account.credit or against_account.credit_in_account_currency
|
||||||
|
credit = against_account.debit or against_account.debit_in_account_currency
|
||||||
|
gl_dict = gl_dict.copy()
|
||||||
|
gl_dict.update(
|
||||||
|
{
|
||||||
|
"against_type": against_account.party_type or "Account",
|
||||||
|
"against": against_account.party or against_account.account,
|
||||||
|
"against_link": against_account.party or against_account.account,
|
||||||
|
"debit": flt(debit, d.precision("debit")),
|
||||||
|
"credit": flt(credit, d.precision("credit")),
|
||||||
|
"account_currency": d.account_currency,
|
||||||
|
"debit_in_account_currency": flt(
|
||||||
|
debit / d.exchange_rate, d.precision("debit_in_account_currency")
|
||||||
|
),
|
||||||
|
"credit_in_account_currency": flt(
|
||||||
|
credit / d.exchange_rate, d.precision("credit_in_account_currency")
|
||||||
|
),
|
||||||
|
}
|
||||||
|
)
|
||||||
|
gl_map.append(gl_dict)
|
||||||
|
|
||||||
return gl_map
|
return gl_map
|
||||||
|
|
||||||
|
def get_transaction_currency_map(self):
|
||||||
|
transaction_currency_map = {}
|
||||||
|
for account in self.get("accounts"):
|
||||||
|
transaction_currency_map.setdefault(account.party or account.account, account.account_currency)
|
||||||
|
|
||||||
|
return transaction_currency_map
|
||||||
|
|
||||||
|
def get_conversion_rate_map(self):
|
||||||
|
conversion_rate_map = {}
|
||||||
|
for account in self.get("accounts"):
|
||||||
|
conversion_rate_map.setdefault(account.party or account.account, account.exchange_rate)
|
||||||
|
|
||||||
|
return conversion_rate_map
|
||||||
|
|
||||||
def make_gl_entries(self, cancel=0, adv_adj=0):
|
def make_gl_entries(self, cancel=0, adv_adj=0):
|
||||||
from erpnext.accounts.general_ledger import make_gl_entries
|
from erpnext.accounts.general_ledger import make_gl_entries
|
||||||
|
|
||||||
@@ -1176,6 +1265,21 @@ class JournalEntry(AccountsController):
|
|||||||
if not self.get("accounts"):
|
if not self.get("accounts"):
|
||||||
frappe.throw(_("Accounts table cannot be blank."))
|
frappe.throw(_("Accounts table cannot be blank."))
|
||||||
|
|
||||||
|
def set_account_and_party_balance(self):
|
||||||
|
account_balance = {}
|
||||||
|
party_balance = {}
|
||||||
|
for d in self.get("accounts"):
|
||||||
|
if d.account not in account_balance:
|
||||||
|
account_balance[d.account] = get_balance_on(account=d.account, date=self.posting_date)
|
||||||
|
|
||||||
|
if (d.party_type, d.party) not in party_balance:
|
||||||
|
party_balance[(d.party_type, d.party)] = get_balance_on(
|
||||||
|
party_type=d.party_type, party=d.party, date=self.posting_date, company=self.company
|
||||||
|
)
|
||||||
|
|
||||||
|
d.account_balance = account_balance[d.account]
|
||||||
|
d.party_balance = party_balance[(d.party_type, d.party)]
|
||||||
|
|
||||||
|
|
||||||
@frappe.whitelist()
|
@frappe.whitelist()
|
||||||
def get_default_bank_cash_account(company, account_type=None, mode_of_payment=None, account=None):
|
def get_default_bank_cash_account(company, account_type=None, mode_of_payment=None, account=None):
|
||||||
@@ -1341,6 +1445,8 @@ def get_payment_entry(ref_doc, args):
|
|||||||
"account_type": frappe.get_cached_value("Account", args.get("party_account"), "account_type"),
|
"account_type": frappe.get_cached_value("Account", args.get("party_account"), "account_type"),
|
||||||
"account_currency": args.get("party_account_currency")
|
"account_currency": args.get("party_account_currency")
|
||||||
or get_account_currency(args.get("party_account")),
|
or get_account_currency(args.get("party_account")),
|
||||||
|
"balance": get_balance_on(args.get("party_account")),
|
||||||
|
"party_balance": get_balance_on(party=args.get("party"), party_type=args.get("party_type")),
|
||||||
"exchange_rate": exchange_rate,
|
"exchange_rate": exchange_rate,
|
||||||
args.get("amount_field_party"): args.get("amount"),
|
args.get("amount_field_party"): args.get("amount"),
|
||||||
"is_advance": args.get("is_advance"),
|
"is_advance": args.get("is_advance"),
|
||||||
@@ -1488,23 +1594,30 @@ def get_outstanding(args):
|
|||||||
|
|
||||||
|
|
||||||
@frappe.whitelist()
|
@frappe.whitelist()
|
||||||
def get_party_account_and_currency(company, party_type, party):
|
def get_party_account_and_balance(company, party_type, party, cost_center=None):
|
||||||
if not frappe.has_permission("Account"):
|
if not frappe.has_permission("Account"):
|
||||||
frappe.msgprint(_("No Permission"), raise_exception=1)
|
frappe.msgprint(_("No Permission"), raise_exception=1)
|
||||||
|
|
||||||
account = get_party_account(party_type, party, company)
|
account = get_party_account(party_type, party, company)
|
||||||
|
|
||||||
|
account_balance = get_balance_on(account=account, cost_center=cost_center)
|
||||||
|
party_balance = get_balance_on(
|
||||||
|
party_type=party_type, party=party, company=company, cost_center=cost_center
|
||||||
|
)
|
||||||
|
|
||||||
return {
|
return {
|
||||||
"account": account,
|
"account": account,
|
||||||
|
"balance": account_balance,
|
||||||
|
"party_balance": party_balance,
|
||||||
"account_currency": frappe.get_cached_value("Account", account, "account_currency"),
|
"account_currency": frappe.get_cached_value("Account", account, "account_currency"),
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@frappe.whitelist()
|
@frappe.whitelist()
|
||||||
def get_account_details_and_party_type(
|
def get_account_balance_and_party_type(
|
||||||
account, date, company, debit=None, credit=None, exchange_rate=None
|
account, date, company, debit=None, credit=None, exchange_rate=None, cost_center=None
|
||||||
):
|
):
|
||||||
"""Returns dict of account details and party type to be set in Journal Entry on selection of account."""
|
"""Returns dict of account balance and party type to be set in Journal Entry on selection of account."""
|
||||||
if not frappe.has_permission("Account"):
|
if not frappe.has_permission("Account"):
|
||||||
frappe.msgprint(_("No Permission"), raise_exception=1)
|
frappe.msgprint(_("No Permission"), raise_exception=1)
|
||||||
|
|
||||||
@@ -1524,6 +1637,7 @@ def get_account_details_and_party_type(
|
|||||||
party_type = ""
|
party_type = ""
|
||||||
|
|
||||||
grid_values = {
|
grid_values = {
|
||||||
|
"balance": get_balance_on(account, date, cost_center=cost_center),
|
||||||
"party_type": party_type,
|
"party_type": party_type,
|
||||||
"account_type": account_details.account_type,
|
"account_type": account_details.account_type,
|
||||||
"account_currency": account_details.account_currency or company_currency,
|
"account_currency": account_details.account_currency or company_currency,
|
||||||
@@ -1641,3 +1755,10 @@ def make_reverse_journal_entry(source_name, target_doc=None):
|
|||||||
)
|
)
|
||||||
|
|
||||||
return doclist
|
return doclist
|
||||||
|
|
||||||
|
|
||||||
|
@frappe.whitelist()
|
||||||
|
def get_against_type(doctype, txt, searchfield, start, page_len, filters):
|
||||||
|
against_types = frappe.db.get_list("Party Type", pluck="name") + ["Account"]
|
||||||
|
doctype = frappe.qb.DocType("DocType")
|
||||||
|
return frappe.qb.from_(doctype).select(doctype.name).where(doctype.name.isin(against_types)).run()
|
||||||
|
|||||||
@@ -1,12 +1,12 @@
|
|||||||
frappe.listview_settings["Journal Entry"] = {
|
frappe.listview_settings['Journal Entry'] = {
|
||||||
add_fields: ["voucher_type", "posting_date", "total_debit", "company", "user_remark"],
|
add_fields: ["voucher_type", "posting_date", "total_debit", "company", "user_remark"],
|
||||||
get_indicator: function (doc) {
|
get_indicator: function(doc) {
|
||||||
if (doc.docstatus == 0) {
|
if(doc.docstatus==0) {
|
||||||
return [__("Draft", "red", "docstatus,=,0")];
|
return [__("Draft", "red", "docstatus,=,0")]
|
||||||
} else if (doc.docstatus == 2) {
|
} else if(doc.docstatus==2) {
|
||||||
return [__("Cancelled", "grey", "docstatus,=,2")];
|
return [__("Cancelled", "grey", "docstatus,=,2")]
|
||||||
} else {
|
} else {
|
||||||
return [__(doc.voucher_type), "blue", "voucher_type,=," + doc.voucher_type];
|
return [__(doc.voucher_type), "blue", "voucher_type,=," + doc.voucher_type]
|
||||||
}
|
}
|
||||||
},
|
}
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -166,37 +166,43 @@ class TestJournalEntry(unittest.TestCase):
|
|||||||
jv.get("accounts")[1].credit_in_account_currency = 5000
|
jv.get("accounts")[1].credit_in_account_currency = 5000
|
||||||
jv.submit()
|
jv.submit()
|
||||||
|
|
||||||
self.voucher_no = jv.name
|
gl_entries = frappe.db.sql(
|
||||||
|
"""select account, account_currency, debit, credit,
|
||||||
|
debit_in_account_currency, credit_in_account_currency
|
||||||
|
from `tabGL Entry` where voucher_type='Journal Entry' and voucher_no=%s
|
||||||
|
order by account asc""",
|
||||||
|
jv.name,
|
||||||
|
as_dict=1,
|
||||||
|
)
|
||||||
|
|
||||||
self.fields = [
|
self.assertTrue(gl_entries)
|
||||||
"account",
|
|
||||||
"account_currency",
|
|
||||||
"debit",
|
|
||||||
"debit_in_account_currency",
|
|
||||||
"credit",
|
|
||||||
"credit_in_account_currency",
|
|
||||||
]
|
|
||||||
|
|
||||||
self.expected_gle = [
|
expected_values = {
|
||||||
{
|
"_Test Bank USD - _TC": {
|
||||||
"account": "_Test Bank - _TC",
|
|
||||||
"account_currency": "INR",
|
|
||||||
"debit": 0,
|
|
||||||
"debit_in_account_currency": 0,
|
|
||||||
"credit": 5000,
|
|
||||||
"credit_in_account_currency": 5000,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"account": "_Test Bank USD - _TC",
|
|
||||||
"account_currency": "USD",
|
"account_currency": "USD",
|
||||||
"debit": 5000,
|
"debit": 5000,
|
||||||
"debit_in_account_currency": 100,
|
"debit_in_account_currency": 100,
|
||||||
"credit": 0,
|
"credit": 0,
|
||||||
"credit_in_account_currency": 0,
|
"credit_in_account_currency": 0,
|
||||||
},
|
},
|
||||||
]
|
"_Test Bank - _TC": {
|
||||||
|
"account_currency": "INR",
|
||||||
|
"debit": 0,
|
||||||
|
"debit_in_account_currency": 0,
|
||||||
|
"credit": 5000,
|
||||||
|
"credit_in_account_currency": 5000,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
self.check_gl_entries()
|
for field in (
|
||||||
|
"account_currency",
|
||||||
|
"debit",
|
||||||
|
"debit_in_account_currency",
|
||||||
|
"credit",
|
||||||
|
"credit_in_account_currency",
|
||||||
|
):
|
||||||
|
for i, gle in enumerate(gl_entries):
|
||||||
|
self.assertEqual(expected_values[gle.account][field], gle[field])
|
||||||
|
|
||||||
# cancel
|
# cancel
|
||||||
jv.cancel()
|
jv.cancel()
|
||||||
@@ -222,37 +228,43 @@ class TestJournalEntry(unittest.TestCase):
|
|||||||
rjv.posting_date = nowdate()
|
rjv.posting_date = nowdate()
|
||||||
rjv.submit()
|
rjv.submit()
|
||||||
|
|
||||||
self.voucher_no = rjv.name
|
gl_entries = frappe.db.sql(
|
||||||
|
"""select account, account_currency, debit, credit,
|
||||||
|
debit_in_account_currency, credit_in_account_currency
|
||||||
|
from `tabGL Entry` where voucher_type='Journal Entry' and voucher_no=%s
|
||||||
|
order by account asc""",
|
||||||
|
rjv.name,
|
||||||
|
as_dict=1,
|
||||||
|
)
|
||||||
|
|
||||||
self.fields = [
|
self.assertTrue(gl_entries)
|
||||||
"account",
|
|
||||||
"account_currency",
|
|
||||||
"debit",
|
|
||||||
"credit",
|
|
||||||
"debit_in_account_currency",
|
|
||||||
"credit_in_account_currency",
|
|
||||||
]
|
|
||||||
|
|
||||||
self.expected_gle = [
|
expected_values = {
|
||||||
{
|
"_Test Bank USD - _TC": {
|
||||||
"account": "_Test Bank USD - _TC",
|
|
||||||
"account_currency": "USD",
|
"account_currency": "USD",
|
||||||
"debit": 0,
|
"debit": 0,
|
||||||
"debit_in_account_currency": 0,
|
"debit_in_account_currency": 0,
|
||||||
"credit": 5000,
|
"credit": 5000,
|
||||||
"credit_in_account_currency": 100,
|
"credit_in_account_currency": 100,
|
||||||
},
|
},
|
||||||
{
|
"Sales - _TC": {
|
||||||
"account": "Sales - _TC",
|
|
||||||
"account_currency": "INR",
|
"account_currency": "INR",
|
||||||
"debit": 5000,
|
"debit": 5000,
|
||||||
"debit_in_account_currency": 5000,
|
"debit_in_account_currency": 5000,
|
||||||
"credit": 0,
|
"credit": 0,
|
||||||
"credit_in_account_currency": 0,
|
"credit_in_account_currency": 0,
|
||||||
},
|
},
|
||||||
]
|
}
|
||||||
|
|
||||||
self.check_gl_entries()
|
for field in (
|
||||||
|
"account_currency",
|
||||||
|
"debit",
|
||||||
|
"debit_in_account_currency",
|
||||||
|
"credit",
|
||||||
|
"credit_in_account_currency",
|
||||||
|
):
|
||||||
|
for i, gle in enumerate(gl_entries):
|
||||||
|
self.assertEqual(expected_values[gle.account][field], gle[field])
|
||||||
|
|
||||||
def test_disallow_change_in_account_currency_for_a_party(self):
|
def test_disallow_change_in_account_currency_for_a_party(self):
|
||||||
# create jv in USD
|
# create jv in USD
|
||||||
@@ -332,25 +344,23 @@ class TestJournalEntry(unittest.TestCase):
|
|||||||
jv.insert()
|
jv.insert()
|
||||||
jv.submit()
|
jv.submit()
|
||||||
|
|
||||||
self.voucher_no = jv.name
|
expected_values = {
|
||||||
|
"_Test Cash - _TC": {"cost_center": cost_center},
|
||||||
|
"_Test Bank - _TC": {"cost_center": cost_center},
|
||||||
|
}
|
||||||
|
|
||||||
self.fields = [
|
gl_entries = frappe.db.sql(
|
||||||
"account",
|
"""select account, cost_center, debit, credit
|
||||||
"cost_center",
|
from `tabGL Entry` where voucher_type='Journal Entry' and voucher_no=%s
|
||||||
]
|
order by account asc""",
|
||||||
|
jv.name,
|
||||||
|
as_dict=1,
|
||||||
|
)
|
||||||
|
|
||||||
self.expected_gle = [
|
self.assertTrue(gl_entries)
|
||||||
{
|
|
||||||
"account": "_Test Bank - _TC",
|
|
||||||
"cost_center": cost_center,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"account": "_Test Cash - _TC",
|
|
||||||
"cost_center": cost_center,
|
|
||||||
},
|
|
||||||
]
|
|
||||||
|
|
||||||
self.check_gl_entries()
|
for gle in gl_entries:
|
||||||
|
self.assertEqual(expected_values[gle.account]["cost_center"], gle.cost_center)
|
||||||
|
|
||||||
def test_jv_with_project(self):
|
def test_jv_with_project(self):
|
||||||
from erpnext.projects.doctype.project.test_project import make_project
|
from erpnext.projects.doctype.project.test_project import make_project
|
||||||
@@ -377,22 +387,23 @@ class TestJournalEntry(unittest.TestCase):
|
|||||||
jv.insert()
|
jv.insert()
|
||||||
jv.submit()
|
jv.submit()
|
||||||
|
|
||||||
self.voucher_no = jv.name
|
expected_values = {
|
||||||
|
"_Test Cash - _TC": {"project": project_name},
|
||||||
|
"_Test Bank - _TC": {"project": project_name},
|
||||||
|
}
|
||||||
|
|
||||||
self.fields = ["account", "project"]
|
gl_entries = frappe.db.sql(
|
||||||
|
"""select account, project, debit, credit
|
||||||
|
from `tabGL Entry` where voucher_type='Journal Entry' and voucher_no=%s
|
||||||
|
order by account asc""",
|
||||||
|
jv.name,
|
||||||
|
as_dict=1,
|
||||||
|
)
|
||||||
|
|
||||||
self.expected_gle = [
|
self.assertTrue(gl_entries)
|
||||||
{
|
|
||||||
"account": "_Test Bank - _TC",
|
|
||||||
"project": project_name,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"account": "_Test Cash - _TC",
|
|
||||||
"project": project_name,
|
|
||||||
},
|
|
||||||
]
|
|
||||||
|
|
||||||
self.check_gl_entries()
|
for gle in gl_entries:
|
||||||
|
self.assertEqual(expected_values[gle.account]["project"], gle.project)
|
||||||
|
|
||||||
def test_jv_account_and_party_balance_with_cost_centre(self):
|
def test_jv_account_and_party_balance_with_cost_centre(self):
|
||||||
from erpnext.accounts.doctype.cost_center.test_cost_center import create_cost_center
|
from erpnext.accounts.doctype.cost_center.test_cost_center import create_cost_center
|
||||||
@@ -415,79 +426,6 @@ class TestJournalEntry(unittest.TestCase):
|
|||||||
account_balance = get_balance_on(account="_Test Bank - _TC", cost_center=cost_center)
|
account_balance = get_balance_on(account="_Test Bank - _TC", cost_center=cost_center)
|
||||||
self.assertEqual(expected_account_balance, account_balance)
|
self.assertEqual(expected_account_balance, account_balance)
|
||||||
|
|
||||||
def test_repost_accounting_entries(self):
|
|
||||||
from erpnext.accounts.doctype.cost_center.test_cost_center import create_cost_center
|
|
||||||
|
|
||||||
# Configure Repost Accounting Ledger for JVs
|
|
||||||
settings = frappe.get_doc("Repost Accounting Ledger Settings")
|
|
||||||
if not [x for x in settings.allowed_types if x.document_type == "Journal Entry"]:
|
|
||||||
settings.append("allowed_types", {"document_type": "Journal Entry", "allowed": True})
|
|
||||||
settings.save()
|
|
||||||
|
|
||||||
# Create JV with defaut cost center - _Test Cost Center
|
|
||||||
jv = make_journal_entry("_Test Cash - _TC", "_Test Bank - _TC", 100, save=False)
|
|
||||||
jv.multi_currency = 0
|
|
||||||
jv.submit()
|
|
||||||
|
|
||||||
# Check GL entries before reposting
|
|
||||||
self.voucher_no = jv.name
|
|
||||||
|
|
||||||
self.fields = [
|
|
||||||
"account",
|
|
||||||
"debit_in_account_currency",
|
|
||||||
"credit_in_account_currency",
|
|
||||||
"cost_center",
|
|
||||||
]
|
|
||||||
|
|
||||||
self.expected_gle = [
|
|
||||||
{
|
|
||||||
"account": "_Test Bank - _TC",
|
|
||||||
"debit_in_account_currency": 0,
|
|
||||||
"credit_in_account_currency": 100,
|
|
||||||
"cost_center": "_Test Cost Center - _TC",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"account": "_Test Cash - _TC",
|
|
||||||
"debit_in_account_currency": 100,
|
|
||||||
"credit_in_account_currency": 0,
|
|
||||||
"cost_center": "_Test Cost Center - _TC",
|
|
||||||
},
|
|
||||||
]
|
|
||||||
|
|
||||||
self.check_gl_entries()
|
|
||||||
|
|
||||||
# Change cost center for bank account - _Test Cost Center for BS Account
|
|
||||||
create_cost_center(cost_center_name="_Test Cost Center for BS Account", company="_Test Company")
|
|
||||||
jv.accounts[1].cost_center = "_Test Cost Center for BS Account - _TC"
|
|
||||||
jv.save()
|
|
||||||
|
|
||||||
# Check if repost flag gets set on update after submit
|
|
||||||
self.assertTrue(jv.repost_required)
|
|
||||||
jv.repost_accounting_entries()
|
|
||||||
|
|
||||||
# Check GL entries after reposting
|
|
||||||
jv.load_from_db()
|
|
||||||
self.expected_gle[0]["cost_center"] = "_Test Cost Center for BS Account - _TC"
|
|
||||||
self.check_gl_entries()
|
|
||||||
|
|
||||||
def check_gl_entries(self):
|
|
||||||
gl = frappe.qb.DocType("GL Entry")
|
|
||||||
query = frappe.qb.from_(gl)
|
|
||||||
for field in self.fields:
|
|
||||||
query = query.select(gl[field])
|
|
||||||
|
|
||||||
query = query.where(
|
|
||||||
(gl.voucher_type == "Journal Entry")
|
|
||||||
& (gl.voucher_no == self.voucher_no)
|
|
||||||
& (gl.is_cancelled == 0)
|
|
||||||
).orderby(gl.account)
|
|
||||||
|
|
||||||
gl_entries = query.run(as_dict=True)
|
|
||||||
|
|
||||||
for i in range(len(self.expected_gle)):
|
|
||||||
for field in self.fields:
|
|
||||||
self.assertEqual(self.expected_gle[i][field], gl_entries[i][field])
|
|
||||||
|
|
||||||
|
|
||||||
def make_journal_entry(
|
def make_journal_entry(
|
||||||
account1,
|
account1,
|
||||||
|
|||||||
@@ -9,10 +9,12 @@
|
|||||||
"field_order": [
|
"field_order": [
|
||||||
"account",
|
"account",
|
||||||
"account_type",
|
"account_type",
|
||||||
|
"balance",
|
||||||
"col_break1",
|
"col_break1",
|
||||||
"bank_account",
|
"bank_account",
|
||||||
"party_type",
|
"party_type",
|
||||||
"party",
|
"party",
|
||||||
|
"party_balance",
|
||||||
"accounting_dimensions_section",
|
"accounting_dimensions_section",
|
||||||
"cost_center",
|
"cost_center",
|
||||||
"dimension_col_break",
|
"dimension_col_break",
|
||||||
@@ -35,7 +37,9 @@
|
|||||||
"col_break3",
|
"col_break3",
|
||||||
"is_advance",
|
"is_advance",
|
||||||
"user_remark",
|
"user_remark",
|
||||||
"against_account"
|
"against_type",
|
||||||
|
"against_account",
|
||||||
|
"against_account_link"
|
||||||
],
|
],
|
||||||
"fields": [
|
"fields": [
|
||||||
{
|
{
|
||||||
@@ -62,7 +66,17 @@
|
|||||||
"print_hide": 1
|
"print_hide": 1
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"allow_on_submit": 1,
|
"fieldname": "balance",
|
||||||
|
"fieldtype": "Currency",
|
||||||
|
"label": "Account Balance",
|
||||||
|
"no_copy": 1,
|
||||||
|
"oldfieldname": "balance",
|
||||||
|
"oldfieldtype": "Data",
|
||||||
|
"options": "account_currency",
|
||||||
|
"print_hide": 1,
|
||||||
|
"read_only": 1
|
||||||
|
},
|
||||||
|
{
|
||||||
"default": ":Company",
|
"default": ":Company",
|
||||||
"description": "If Income or Expense",
|
"description": "If Income or Expense",
|
||||||
"fieldname": "cost_center",
|
"fieldname": "cost_center",
|
||||||
@@ -95,6 +109,14 @@
|
|||||||
"label": "Party",
|
"label": "Party",
|
||||||
"options": "party_type"
|
"options": "party_type"
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "party_balance",
|
||||||
|
"fieldtype": "Currency",
|
||||||
|
"label": "Party Balance",
|
||||||
|
"options": "account_currency",
|
||||||
|
"print_hide": 1,
|
||||||
|
"read_only": 1
|
||||||
|
},
|
||||||
{
|
{
|
||||||
"fieldname": "currency_section",
|
"fieldname": "currency_section",
|
||||||
"fieldtype": "Section Break",
|
"fieldtype": "Section Break",
|
||||||
@@ -203,7 +225,6 @@
|
|||||||
"no_copy": 1
|
"no_copy": 1
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"allow_on_submit": 1,
|
|
||||||
"fieldname": "project",
|
"fieldname": "project",
|
||||||
"fieldtype": "Link",
|
"fieldtype": "Link",
|
||||||
"label": "Project",
|
"label": "Project",
|
||||||
@@ -231,14 +252,21 @@
|
|||||||
"print_hide": 1
|
"print_hide": 1
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"fieldname": "against_account",
|
"fieldname": "against_account",
|
||||||
"fieldtype": "Text",
|
"fieldtype": "Text",
|
||||||
"hidden": 1,
|
"hidden": 1,
|
||||||
|
"label": "Against Account",
|
||||||
|
"no_copy": 1,
|
||||||
|
"oldfieldname": "against_account",
|
||||||
|
"oldfieldtype": "Text",
|
||||||
|
"print_hide": 1
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "against_account_link",
|
||||||
|
"fieldtype": "Dynamic Link",
|
||||||
"label": "Against Account",
|
"label": "Against Account",
|
||||||
"no_copy": 1,
|
"no_copy": 1,
|
||||||
"oldfieldname": "against_account",
|
"options": "against_type"
|
||||||
"oldfieldtype": "Text",
|
|
||||||
"print_hide": 1
|
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"collapsible": 1,
|
"collapsible": 1,
|
||||||
@@ -261,14 +289,19 @@
|
|||||||
"fieldtype": "Data",
|
"fieldtype": "Data",
|
||||||
"hidden": 1,
|
"hidden": 1,
|
||||||
"label": "Reference Detail No",
|
"label": "Reference Detail No",
|
||||||
"no_copy": 1,
|
"no_copy": 1
|
||||||
"search_index": 1
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "against_type",
|
||||||
|
"fieldtype": "Link",
|
||||||
|
"label": "Against Type",
|
||||||
|
"options": "DocType"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"idx": 1,
|
"idx": 1,
|
||||||
"istable": 1,
|
"istable": 1,
|
||||||
"links": [],
|
"links": [],
|
||||||
"modified": "2024-02-05 01:10:50.224840",
|
"modified": "2023-12-02 23:21:22.205409",
|
||||||
"modified_by": "Administrator",
|
"modified_by": "Administrator",
|
||||||
"module": "Accounts",
|
"module": "Accounts",
|
||||||
"name": "Journal Entry Account",
|
"name": "Journal Entry Account",
|
||||||
|
|||||||
@@ -2,85 +2,78 @@
|
|||||||
// For license information, please see license.txt
|
// For license information, please see license.txt
|
||||||
|
|
||||||
frappe.ui.form.on("Journal Entry Template", {
|
frappe.ui.form.on("Journal Entry Template", {
|
||||||
onload: function (frm) {
|
onload: function(frm) {
|
||||||
if (frm.is_new()) {
|
if(frm.is_new()) {
|
||||||
frappe.call({
|
frappe.call({
|
||||||
type: "GET",
|
type: "GET",
|
||||||
method: "erpnext.accounts.doctype.journal_entry_template.journal_entry_template.get_naming_series",
|
method: "erpnext.accounts.doctype.journal_entry_template.journal_entry_template.get_naming_series",
|
||||||
callback: function (r) {
|
callback: function(r){
|
||||||
if (r.message) {
|
if(r.message) {
|
||||||
frm.set_df_property("naming_series", "options", r.message.split("\n"));
|
frm.set_df_property("naming_series", "options", r.message.split("\n"));
|
||||||
frm.set_value("naming_series", r.message.split("\n")[0]);
|
frm.set_value("naming_series", r.message.split("\n")[0]);
|
||||||
frm.refresh_field("naming_series");
|
frm.refresh_field("naming_series");
|
||||||
}
|
}
|
||||||
},
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
refresh: function (frm) {
|
refresh: function(frm) {
|
||||||
frappe.model.set_default_values(frm.doc);
|
frappe.model.set_default_values(frm.doc);
|
||||||
|
|
||||||
frm.set_query("account", "accounts", function () {
|
frm.set_query("account" ,"accounts", function(){
|
||||||
var filters = {
|
var filters = {
|
||||||
company: frm.doc.company,
|
company: frm.doc.company,
|
||||||
is_group: 0,
|
is_group: 0
|
||||||
};
|
};
|
||||||
|
|
||||||
if (!frm.doc.multi_currency) {
|
if(!frm.doc.multi_currency) {
|
||||||
$.extend(filters, {
|
$.extend(filters, {
|
||||||
account_currency: [
|
account_currency: ['in', [frappe.get_doc(":Company", frm.doc.company).default_currency, null]]
|
||||||
"in",
|
|
||||||
[frappe.get_doc(":Company", frm.doc.company).default_currency, null],
|
|
||||||
],
|
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
return { filters: filters };
|
return { filters: filters };
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
voucher_type: function (frm) {
|
voucher_type: function(frm) {
|
||||||
var add_accounts = function (doc, r) {
|
var add_accounts = function(doc, r) {
|
||||||
$.each(r, function (i, d) {
|
$.each(r, function(i, d) {
|
||||||
var row = frappe.model.add_child(doc, "Journal Entry Template Account", "accounts");
|
var row = frappe.model.add_child(doc, "Journal Entry Template Account", "accounts");
|
||||||
row.account = d.account;
|
row.account = d.account;
|
||||||
});
|
});
|
||||||
refresh_field("accounts");
|
refresh_field("accounts");
|
||||||
};
|
};
|
||||||
|
|
||||||
if (!frm.doc.company) return;
|
if(!frm.doc.company) return;
|
||||||
|
|
||||||
frm.trigger("clear_child");
|
frm.trigger("clear_child");
|
||||||
switch (frm.doc.voucher_type) {
|
switch(frm.doc.voucher_type){
|
||||||
case "Bank Entry":
|
case "Bank Entry":
|
||||||
case "Cash Entry":
|
case "Cash Entry":
|
||||||
frappe.call({
|
frappe.call({
|
||||||
type: "GET",
|
type: "GET",
|
||||||
method: "erpnext.accounts.doctype.journal_entry.journal_entry.get_default_bank_cash_account",
|
method: "erpnext.accounts.doctype.journal_entry.journal_entry.get_default_bank_cash_account",
|
||||||
args: {
|
args: {
|
||||||
account_type:
|
"account_type": (frm.doc.voucher_type=="Bank Entry" ?
|
||||||
frm.doc.voucher_type == "Bank Entry"
|
"Bank" : (frm.doc.voucher_type=="Cash Entry" ? "Cash" : null)),
|
||||||
? "Bank"
|
"company": frm.doc.company
|
||||||
: frm.doc.voucher_type == "Cash Entry"
|
|
||||||
? "Cash"
|
|
||||||
: null,
|
|
||||||
company: frm.doc.company,
|
|
||||||
},
|
},
|
||||||
callback: function (r) {
|
callback: function(r) {
|
||||||
if (r.message) {
|
if(r.message) {
|
||||||
// If default company bank account not set
|
// If default company bank account not set
|
||||||
if (!$.isEmptyObject(r.message)) {
|
if(!$.isEmptyObject(r.message)){
|
||||||
add_accounts(frm.doc, [r.message]);
|
add_accounts(frm.doc, [r.message]);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
}
|
||||||
});
|
});
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
frm.trigger("clear_child");
|
frm.trigger("clear_child");
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
clear_child: function (frm) {
|
clear_child: function(frm){
|
||||||
frappe.model.clear_table(frm.doc, "accounts");
|
frappe.model.clear_table(frm.doc, "accounts");
|
||||||
frm.refresh_field("accounts");
|
frm.refresh_field("accounts");
|
||||||
},
|
}
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -1,9 +1,9 @@
|
|||||||
// Copyright (c) 2021, Wahni Green Technologies Pvt. Ltd. and contributors
|
// Copyright (c) 2021, Wahni Green Technologies Pvt. Ltd. and contributors
|
||||||
// For license information, please see license.txt
|
// For license information, please see license.txt
|
||||||
|
|
||||||
frappe.ui.form.on("Ledger Merge", {
|
frappe.ui.form.on('Ledger Merge', {
|
||||||
setup: function (frm) {
|
setup: function(frm) {
|
||||||
frappe.realtime.on("ledger_merge_refresh", ({ ledger_merge }) => {
|
frappe.realtime.on('ledger_merge_refresh', ({ ledger_merge }) => {
|
||||||
if (ledger_merge !== frm.doc.name) return;
|
if (ledger_merge !== frm.doc.name) return;
|
||||||
frappe.model.clear_doc(frm.doc.doctype, frm.doc.name);
|
frappe.model.clear_doc(frm.doc.doctype, frm.doc.name);
|
||||||
frappe.model.with_doc(frm.doc.doctype, frm.doc.name).then(() => {
|
frappe.model.with_doc(frm.doc.doctype, frm.doc.name).then(() => {
|
||||||
@@ -11,29 +11,29 @@ frappe.ui.form.on("Ledger Merge", {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
frappe.realtime.on("ledger_merge_progress", (data) => {
|
frappe.realtime.on('ledger_merge_progress', data => {
|
||||||
if (data.ledger_merge !== frm.doc.name) return;
|
if (data.ledger_merge !== frm.doc.name) return;
|
||||||
let message = __("Merging {0} of {1}", [data.current, data.total]);
|
let message = __('Merging {0} of {1}', [data.current, data.total]);
|
||||||
let percent = Math.floor((data.current * 100) / data.total);
|
let percent = Math.floor((data.current * 100) / data.total);
|
||||||
frm.dashboard.show_progress(__("Merge Progress"), percent, message);
|
frm.dashboard.show_progress(__('Merge Progress'), percent, message);
|
||||||
frm.page.set_indicator(__("In Progress"), "orange");
|
frm.page.set_indicator(__('In Progress'), 'orange');
|
||||||
});
|
});
|
||||||
|
|
||||||
frm.set_query("account", function (doc) {
|
frm.set_query("account", function(doc) {
|
||||||
if (!doc.company) frappe.throw(__("Please set Company"));
|
if (!doc.company) frappe.throw(__('Please set Company'));
|
||||||
if (!doc.root_type) frappe.throw(__("Please set Root Type"));
|
if (!doc.root_type) frappe.throw(__('Please set Root Type'));
|
||||||
return {
|
return {
|
||||||
filters: {
|
filters: {
|
||||||
root_type: doc.root_type,
|
root_type: doc.root_type,
|
||||||
company: doc.company,
|
company: doc.company
|
||||||
},
|
}
|
||||||
};
|
};
|
||||||
});
|
});
|
||||||
|
|
||||||
frm.set_query("account", "merge_accounts", function (doc) {
|
frm.set_query('account', 'merge_accounts', function(doc) {
|
||||||
if (!doc.company) frappe.throw(__("Please set Company"));
|
if (!doc.company) frappe.throw(__('Please set Company'));
|
||||||
if (!doc.root_type) frappe.throw(__("Please set Root Type"));
|
if (!doc.root_type) frappe.throw(__('Please set Root Type'));
|
||||||
if (!doc.account) frappe.throw(__("Please set Account"));
|
if (!doc.account) frappe.throw(__('Please set Account'));
|
||||||
let acc = [doc.account];
|
let acc = [doc.account];
|
||||||
frm.doc.merge_accounts.forEach((row) => {
|
frm.doc.merge_accounts.forEach((row) => {
|
||||||
acc.push(row.account);
|
acc.push(row.account);
|
||||||
@@ -43,86 +43,86 @@ frappe.ui.form.on("Ledger Merge", {
|
|||||||
is_group: doc.is_group,
|
is_group: doc.is_group,
|
||||||
root_type: doc.root_type,
|
root_type: doc.root_type,
|
||||||
name: ["not in", acc],
|
name: ["not in", acc],
|
||||||
company: doc.company,
|
company: doc.company
|
||||||
},
|
}
|
||||||
};
|
};
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
|
|
||||||
refresh: function (frm) {
|
refresh: function(frm) {
|
||||||
frm.page.hide_icon_group();
|
frm.page.hide_icon_group();
|
||||||
frm.trigger("set_merge_status");
|
frm.trigger('set_merge_status');
|
||||||
frm.trigger("update_primary_action");
|
frm.trigger('update_primary_action');
|
||||||
},
|
},
|
||||||
|
|
||||||
after_save: function (frm) {
|
after_save: function(frm) {
|
||||||
setTimeout(() => {
|
setTimeout(() => {
|
||||||
frm.trigger("update_primary_action");
|
frm.trigger('update_primary_action');
|
||||||
}, 500);
|
}, 500);
|
||||||
},
|
},
|
||||||
|
|
||||||
update_primary_action: function (frm) {
|
update_primary_action: function(frm) {
|
||||||
if (frm.is_dirty()) {
|
if (frm.is_dirty()) {
|
||||||
frm.enable_save();
|
frm.enable_save();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
frm.disable_save();
|
frm.disable_save();
|
||||||
if (frm.doc.status !== "Success") {
|
if (frm.doc.status !== 'Success') {
|
||||||
if (!frm.is_new()) {
|
if (!frm.is_new()) {
|
||||||
let label = frm.doc.status === "Pending" ? __("Start Merge") : __("Retry");
|
let label = frm.doc.status === 'Pending' ? __('Start Merge') : __('Retry');
|
||||||
frm.page.set_primary_action(label, () => frm.events.start_merge(frm));
|
frm.page.set_primary_action(label, () => frm.events.start_merge(frm));
|
||||||
} else {
|
} else {
|
||||||
frm.page.set_primary_action(__("Save"), () => frm.save());
|
frm.page.set_primary_action(__('Save'), () => frm.save());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
start_merge: function (frm) {
|
start_merge: function(frm) {
|
||||||
frm.call({
|
frm.call({
|
||||||
method: "form_start_merge",
|
method: 'form_start_merge',
|
||||||
args: { docname: frm.doc.name },
|
args: { docname: frm.doc.name },
|
||||||
btn: frm.page.btn_primary,
|
btn: frm.page.btn_primary
|
||||||
}).then((r) => {
|
}).then(r => {
|
||||||
if (r.message === true) {
|
if (r.message === true) {
|
||||||
frm.disable_save();
|
frm.disable_save();
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
|
|
||||||
set_merge_status: function (frm) {
|
set_merge_status: function(frm) {
|
||||||
if (frm.doc.status == "Pending") return;
|
if (frm.doc.status == "Pending") return;
|
||||||
let successful_records = 0;
|
let successful_records = 0;
|
||||||
frm.doc.merge_accounts.forEach((row) => {
|
frm.doc.merge_accounts.forEach((row) => {
|
||||||
if (row.merged) successful_records += 1;
|
if (row.merged) successful_records += 1;
|
||||||
});
|
});
|
||||||
let message_args = [successful_records, frm.doc.merge_accounts.length];
|
let message_args = [successful_records, frm.doc.merge_accounts.length];
|
||||||
frm.dashboard.set_headline(__("Successfully merged {0} out of {1}.", message_args));
|
frm.dashboard.set_headline(__('Successfully merged {0} out of {1}.', message_args));
|
||||||
},
|
},
|
||||||
|
|
||||||
root_type: function (frm) {
|
root_type: function(frm) {
|
||||||
frm.set_value("account", "");
|
frm.set_value('account', '');
|
||||||
frm.set_value("merge_accounts", []);
|
frm.set_value('merge_accounts', []);
|
||||||
},
|
},
|
||||||
|
|
||||||
company: function (frm) {
|
company: function(frm) {
|
||||||
frm.set_value("account", "");
|
frm.set_value('account', '');
|
||||||
frm.set_value("merge_accounts", []);
|
frm.set_value('merge_accounts', []);
|
||||||
},
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
frappe.ui.form.on("Ledger Merge Accounts", {
|
frappe.ui.form.on('Ledger Merge Accounts', {
|
||||||
merge_accounts_add: function (frm) {
|
merge_accounts_add: function(frm) {
|
||||||
frm.trigger("update_primary_action");
|
frm.trigger('update_primary_action');
|
||||||
},
|
},
|
||||||
|
|
||||||
merge_accounts_remove: function (frm) {
|
merge_accounts_remove: function(frm) {
|
||||||
frm.trigger("update_primary_action");
|
frm.trigger('update_primary_action');
|
||||||
},
|
},
|
||||||
|
|
||||||
account: function (frm, cdt, cdn) {
|
account: function(frm, cdt, cdn) {
|
||||||
let row = frappe.get_doc(cdt, cdn);
|
let row = frappe.get_doc(cdt, cdn);
|
||||||
row.account_name = row.account;
|
row.account_name = row.account;
|
||||||
frm.refresh_field("merge_accounts");
|
frm.refresh_field('merge_accounts');
|
||||||
frm.trigger("update_primary_action");
|
frm.trigger('update_primary_action');
|
||||||
},
|
}
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -1,6 +1,8 @@
|
|||||||
// Copyright (c) 2018, Frappe Technologies Pvt. Ltd. and contributors
|
// Copyright (c) 2018, Frappe Technologies Pvt. Ltd. and contributors
|
||||||
// For license information, please see license.txt
|
// For license information, please see license.txt
|
||||||
|
|
||||||
frappe.ui.form.on("Loyalty Point Entry", {
|
frappe.ui.form.on('Loyalty Point Entry', {
|
||||||
refresh: function (frm) {},
|
refresh: function(frm) {
|
||||||
|
|
||||||
|
}
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -3,37 +3,30 @@
|
|||||||
|
|
||||||
frappe.provide("erpnext.accounts.dimensions");
|
frappe.provide("erpnext.accounts.dimensions");
|
||||||
|
|
||||||
frappe.ui.form.on("Loyalty Program", {
|
frappe.ui.form.on('Loyalty Program', {
|
||||||
setup: function (frm) {
|
setup: function(frm) {
|
||||||
var help_content = `<table class="table table-bordered" style="background-color: var(--scrollbar-track-color);">
|
var help_content =
|
||||||
|
`<table class="table table-bordered" style="background-color: var(--scrollbar-track-color);">
|
||||||
<tr><td>
|
<tr><td>
|
||||||
<h4>
|
<h4>
|
||||||
<i class="fa fa-hand-right"></i>
|
<i class="fa fa-hand-right"></i>
|
||||||
${__("Notes")}
|
${__('Notes')}
|
||||||
</h4>
|
</h4>
|
||||||
<ul>
|
<ul>
|
||||||
<li>
|
<li>
|
||||||
${__(
|
${__("Loyalty Points will be calculated from the spent done (via the Sales Invoice), based on collection factor mentioned.")}
|
||||||
"Loyalty Points will be calculated from the spent done (via the Sales Invoice), based on collection factor mentioned."
|
|
||||||
)}
|
|
||||||
</li>
|
</li>
|
||||||
<li>
|
<li>
|
||||||
${__(
|
${__("There can be multiple tiered collection factor based on the total spent. But the conversion factor for redemption will always be same for all the tier.")}
|
||||||
"There can be multiple tiered collection factor based on the total spent. But the conversion factor for redemption will always be same for all the tier."
|
|
||||||
)}
|
|
||||||
</li>
|
</li>
|
||||||
<li>
|
<li>
|
||||||
${__(
|
${__("In the case of multi-tier program, Customers will be auto assigned to the concerned tier as per their spent")}
|
||||||
"In the case of multi-tier program, Customers will be auto assigned to the concerned tier as per their spent"
|
|
||||||
)}
|
|
||||||
</li>
|
</li>
|
||||||
<li>
|
<li>
|
||||||
${__("If unlimited expiry for the Loyalty Points, keep the Expiry Duration empty or 0.")}
|
${__("If unlimited expiry for the Loyalty Points, keep the Expiry Duration empty or 0.")}
|
||||||
</li>
|
</li>
|
||||||
<li>
|
<li>
|
||||||
${__(
|
${__("If Auto Opt In is checked, then the customers will be automatically linked with the concerned Loyalty Program (on save)")}
|
||||||
"If Auto Opt In is checked, then the customers will be automatically linked with the concerned Loyalty Program (on save)"
|
|
||||||
)}
|
|
||||||
</li>
|
</li>
|
||||||
<li>
|
<li>
|
||||||
${__("One customer can be part of only single Loyalty Program.")}
|
${__("One customer can be part of only single Loyalty Program.")}
|
||||||
@@ -44,14 +37,14 @@ frappe.ui.form.on("Loyalty Program", {
|
|||||||
set_field_options("loyalty_program_help", help_content);
|
set_field_options("loyalty_program_help", help_content);
|
||||||
},
|
},
|
||||||
|
|
||||||
onload: function (frm) {
|
onload: function(frm) {
|
||||||
frm.set_query("expense_account", function (doc) {
|
frm.set_query("expense_account", function(doc) {
|
||||||
return {
|
return {
|
||||||
filters: {
|
filters: {
|
||||||
root_type: "Expense",
|
"root_type": "Expense",
|
||||||
is_group: 0,
|
'is_group': 0,
|
||||||
company: doc.company,
|
'company': doc.company
|
||||||
},
|
}
|
||||||
};
|
};
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -59,15 +52,13 @@ frappe.ui.form.on("Loyalty Program", {
|
|||||||
erpnext.accounts.dimensions.setup_dimension_filters(frm, frm.doctype);
|
erpnext.accounts.dimensions.setup_dimension_filters(frm, frm.doctype);
|
||||||
},
|
},
|
||||||
|
|
||||||
refresh: function (frm) {
|
refresh: function(frm) {
|
||||||
if (frm.doc.loyalty_program_type === "Single Tier Program" && frm.doc.collection_rules.length > 1) {
|
if (frm.doc.loyalty_program_type === "Single Tier Program" && frm.doc.collection_rules.length > 1) {
|
||||||
frappe.throw(
|
frappe.throw(__("Please select the Multiple Tier Program type for more than one collection rules."));
|
||||||
__("Please select the Multiple Tier Program type for more than one collection rules.")
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
company: function (frm) {
|
company: function(frm) {
|
||||||
erpnext.accounts.dimensions.update_dimension(frm, frm.doctype);
|
erpnext.accounts.dimensions.update_dimension(frm, frm.doctype);
|
||||||
},
|
}
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -140,7 +140,7 @@ class TestLoyaltyProgram(unittest.TestCase):
|
|||||||
"Loyalty Point Entry",
|
"Loyalty Point Entry",
|
||||||
{"invoice_type": "Sales Invoice", "invoice": si.name, "customer": si.customer},
|
{"invoice_type": "Sales Invoice", "invoice": si.name, "customer": si.customer},
|
||||||
)
|
)
|
||||||
self.assertEqual(True, not (lpe is None))
|
self.assertEqual(True, lpe is not None)
|
||||||
|
|
||||||
# cancelling sales invoice
|
# cancelling sales invoice
|
||||||
si.cancel()
|
si.cancel()
|
||||||
|
|||||||
@@ -1,16 +1,16 @@
|
|||||||
// Copyright (c) 2015, Frappe Technologies Pvt. Ltd. and Contributors
|
// Copyright (c) 2015, Frappe Technologies Pvt. Ltd. and Contributors
|
||||||
// License: GNU General Public License v3. See license.txt
|
// License: GNU General Public License v3. See license.txt
|
||||||
|
|
||||||
frappe.ui.form.on("Mode of Payment", {
|
frappe.ui.form.on('Mode of Payment', {
|
||||||
setup: function (frm) {
|
setup: function(frm) {
|
||||||
frm.set_query("default_account", "accounts", function (doc, cdt, cdn) {
|
frm.set_query("default_account", "accounts", function(doc, cdt, cdn) {
|
||||||
let d = locals[cdt][cdn];
|
let d = locals[cdt][cdn];
|
||||||
return {
|
return {
|
||||||
filters: [
|
filters: [
|
||||||
["Account", "account_type", "in", "Bank, Cash, Receivable"],
|
['Account', 'account_type', 'in', 'Bank, Cash, Receivable'],
|
||||||
["Account", "is_group", "=", 0],
|
['Account', 'is_group', '=', 0],
|
||||||
["Account", "company", "=", d.company],
|
['Account', 'company', '=', d.company]
|
||||||
],
|
]
|
||||||
};
|
};
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
|
|||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user