mirror of
https://github.com/jumpserver/lina.git
synced 2026-01-14 03:46:26 +00:00
Compare commits
1506 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
7aa40de518 | ||
|
|
17fd374e0a | ||
|
|
df5d15c1e5 | ||
|
|
b58c21a79f | ||
|
|
3d81f92667 | ||
|
|
a6d642df60 | ||
|
|
664855d4b0 | ||
|
|
c79637095a | ||
|
|
2a96abdd4a | ||
|
|
9541b97b23 | ||
|
|
52518a9ff3 | ||
|
|
24a1c11288 | ||
|
|
313aebaf50 | ||
|
|
5c6373e689 | ||
|
|
09fe3ea107 | ||
|
|
ccf081b608 | ||
|
|
80ce3293a1 | ||
|
|
4ed282ff2b | ||
|
|
dd8957cb69 | ||
|
|
36c687b854 | ||
|
|
e22ecb6fe8 | ||
|
|
b77440284f | ||
|
|
93d866328c | ||
|
|
14ba501f2b | ||
|
|
f072546083 | ||
|
|
8f9ebcdfa9 | ||
|
|
7b76917768 | ||
|
|
e4d4bc84b6 | ||
|
|
62cf19e70e | ||
|
|
2cb36f89f0 | ||
|
|
9946ad75ad | ||
|
|
abd8919225 | ||
|
|
6e5e760689 | ||
|
|
1235895973 | ||
|
|
59f9025e42 | ||
|
|
d06bda6d40 | ||
|
|
a8099089b2 | ||
|
|
9a04c81238 | ||
|
|
f1b3e5038f | ||
|
|
f6fe08607b | ||
|
|
6f2ca3e26a | ||
|
|
3ed9ac0d9b | ||
|
|
821ed14f40 | ||
|
|
6f0ee734e5 | ||
|
|
9a5b174eb1 | ||
|
|
c27e1c97f2 | ||
|
|
1b4e01d19e | ||
|
|
28c836b1f6 | ||
|
|
fc85eaf6b9 | ||
|
|
678d17a4e7 | ||
|
|
ddf9780f9c | ||
|
|
d47bd4acad | ||
|
|
09dabb5d3f | ||
|
|
9eea051884 | ||
|
|
f6a3eb1349 | ||
|
|
433f3a34cb | ||
|
|
aa790944f6 | ||
|
|
57167fc821 | ||
|
|
5988892840 | ||
|
|
af11e1bf0c | ||
|
|
09416b13d3 | ||
|
|
478745f534 | ||
|
|
3de2bf73ea | ||
|
|
3b877739a3 | ||
|
|
06899d6932 | ||
|
|
6f3d21bb77 | ||
|
|
b7da517ad7 | ||
|
|
598b6b12e8 | ||
|
|
400ceac737 | ||
|
|
34de6d8775 | ||
|
|
080542633a | ||
|
|
653f26137b | ||
|
|
c86f0cc08d | ||
|
|
541c6c5fe5 | ||
|
|
c36a210cd4 | ||
|
|
b9f26df5e6 | ||
|
|
dd0baa7b00 | ||
|
|
524105278b | ||
|
|
2656546e3b | ||
|
|
ecaf1ceace | ||
|
|
cf515a18de | ||
|
|
a0ab7d3f32 | ||
|
|
467c3f7288 | ||
|
|
f68eaee8c8 | ||
|
|
163f661386 | ||
|
|
e54bc076d6 | ||
|
|
1fbf04ae51 | ||
|
|
f002c4aa3d | ||
|
|
f67bd39067 | ||
|
|
5c60624b2d | ||
|
|
488684f293 | ||
|
|
5c511345bf | ||
|
|
be85a91ccd | ||
|
|
94a02a2e7e | ||
|
|
e60a33a2b1 | ||
|
|
8449dafd55 | ||
|
|
a787737290 | ||
|
|
2e74f1522f | ||
|
|
fa517c8325 | ||
|
|
0748b6ce0c | ||
|
|
59ee3eff17 | ||
|
|
5cc17de1e0 | ||
|
|
fe615e0314 | ||
|
|
42aacd9df5 | ||
|
|
bb27171b09 | ||
|
|
ccc163ef07 | ||
|
|
e63630fce7 | ||
|
|
17748c56c9 | ||
|
|
d880e5cb8c | ||
|
|
310fe8068a | ||
|
|
8670c3988d | ||
|
|
b5144625ee | ||
|
|
58b6d75ccf | ||
|
|
c9ec67cc50 | ||
|
|
e4d1533091 | ||
|
|
97440860d9 | ||
|
|
3dd31dc76e | ||
|
|
1822d72a9b | ||
|
|
3267839327 | ||
|
|
17d73f2b3c | ||
|
|
ec49241272 | ||
|
|
c2e10b8a34 | ||
|
|
29813e7ce1 | ||
|
|
919d95b56b | ||
|
|
5f95fc0422 | ||
|
|
49a4000086 | ||
|
|
9c7818313e | ||
|
|
705e236279 | ||
|
|
d4ecb33c78 | ||
|
|
67e45ad284 | ||
|
|
8cb72c3d82 | ||
|
|
b65a007d37 | ||
|
|
e487ee24af | ||
|
|
bfbc78e342 | ||
|
|
54879a8174 | ||
|
|
36634a2ce5 | ||
|
|
07920a4917 | ||
|
|
25932587a5 | ||
|
|
d98534d039 | ||
|
|
6cb5d6ac46 | ||
|
|
fab461c10d | ||
|
|
644980bc40 | ||
|
|
a56e3843bf | ||
|
|
5ca188c014 | ||
|
|
510cf48def | ||
|
|
e64fdab7fc | ||
|
|
aaec496642 | ||
|
|
80c05c9b65 | ||
|
|
c9159838f5 | ||
|
|
b8e005f4b1 | ||
|
|
6e73ded101 | ||
|
|
6926556fa3 | ||
|
|
24971c1112 | ||
|
|
763792f42e | ||
|
|
0ff28d2626 | ||
|
|
f607eeda42 | ||
|
|
c0145f4ec4 | ||
|
|
7c995c005b | ||
|
|
21f4f003bb | ||
|
|
fbbac6cea5 | ||
|
|
302400e350 | ||
|
|
b2120678dd | ||
|
|
03c7755703 | ||
|
|
21373574de | ||
|
|
342ee31c72 | ||
|
|
7aa8afec44 | ||
|
|
d6b54c9879 | ||
|
|
b1952a180f | ||
|
|
d3a7fb63b9 | ||
|
|
53cfdbd3a1 | ||
|
|
41cd77d75e | ||
|
|
15ac510c6f | ||
|
|
31e005eca7 | ||
|
|
842ea3cbf3 | ||
|
|
9f893af74b | ||
|
|
5274dc4e6b | ||
|
|
64441e4836 | ||
|
|
43997eeac1 | ||
|
|
1b95af54c1 | ||
|
|
e79f95a822 | ||
|
|
fb85e168e7 | ||
|
|
521cafa2af | ||
|
|
bb9f905dd4 | ||
|
|
3eda4381bb | ||
|
|
080acf57f7 | ||
|
|
9c137e6763 | ||
|
|
75f5363602 | ||
|
|
32f6d5dc4a | ||
|
|
21265aa983 | ||
|
|
e11441e5a7 | ||
|
|
e7624f47bf | ||
|
|
7276b19a92 | ||
|
|
32fa172197 | ||
|
|
aa80a07bfc | ||
|
|
fd97a6c4a2 | ||
|
|
9b355a2942 | ||
|
|
0431f7108a | ||
|
|
3500ab89f5 | ||
|
|
fcdcf29efa | ||
|
|
49d74271c8 | ||
|
|
9a7eca3cf8 | ||
|
|
dc686f2af7 | ||
|
|
172c6d69a1 | ||
|
|
3f2a846184 | ||
|
|
48ac797b59 | ||
|
|
92962bdc5a | ||
|
|
ca4889a19a | ||
|
|
e4631a5309 | ||
|
|
62eced61d2 | ||
|
|
7a8df3fb7b | ||
|
|
91ef3cfaab | ||
|
|
c92843e973 | ||
|
|
47372df4d2 | ||
|
|
143e913554 | ||
|
|
61b2b0fb23 | ||
|
|
54ac31920c | ||
|
|
4eaeff0a18 | ||
|
|
08c16aae72 | ||
|
|
bf5e218edc | ||
|
|
e8dbd99f74 | ||
|
|
9ab2acfa5b | ||
|
|
d3cbb48e05 | ||
|
|
170dde2ba5 | ||
|
|
021eb1fd5b | ||
|
|
1b2c85d86d | ||
|
|
09f734e6fc | ||
|
|
2434e434b7 | ||
|
|
3a5aa7bf90 | ||
|
|
dbe4c232c1 | ||
|
|
3294495577 | ||
|
|
05e7e6dd06 | ||
|
|
936de3c9fb | ||
|
|
51dac8ca30 | ||
|
|
89b3ea51f5 | ||
|
|
bdfc608534 | ||
|
|
667514e90b | ||
|
|
83a116c094 | ||
|
|
211933349e | ||
|
|
ec51b243b1 | ||
|
|
80eb072c1b | ||
|
|
ce08fd57c6 | ||
|
|
40ccab1a19 | ||
|
|
8489d94643 | ||
|
|
72693af5fd | ||
|
|
1fdcf9ff75 | ||
|
|
c4ad25bfe3 | ||
|
|
d6a7d93398 | ||
|
|
43b2ad3104 | ||
|
|
2209eea890 | ||
|
|
1a56546d2e | ||
|
|
fa52226958 | ||
|
|
42ee6c0848 | ||
|
|
49f63d638c | ||
|
|
7e6d78d223 | ||
|
|
a99a5a6312 | ||
|
|
e80b162eca | ||
|
|
fabf60ac47 | ||
|
|
14b5eb5239 | ||
|
|
e365eaef8a | ||
|
|
d14cbce2d6 | ||
|
|
f29f88b78b | ||
|
|
40ccff9685 | ||
|
|
b85de1de33 | ||
|
|
640e758016 | ||
|
|
09bd49941f | ||
|
|
edc74c8dfb | ||
|
|
fcd7d17ef1 | ||
|
|
574b7b54c5 | ||
|
|
f486da0018 | ||
|
|
1598b4e4ad | ||
|
|
4556d3f4fe | ||
|
|
e521868cd2 | ||
|
|
d3d6a0e890 | ||
|
|
fd4223e107 | ||
|
|
152301b0f3 | ||
|
|
cb7a492b74 | ||
|
|
80dd8a23ec | ||
|
|
41729ebe53 | ||
|
|
da3e6aff76 | ||
|
|
7ff8114850 | ||
|
|
43eaaf3eba | ||
|
|
26d8154db8 | ||
|
|
673198af07 | ||
|
|
b14630c33a | ||
|
|
187a0824fc | ||
|
|
e180fbd2a5 | ||
|
|
90ebb0ff0b | ||
|
|
c922fa99fd | ||
|
|
5e3917d61c | ||
|
|
57702bcef3 | ||
|
|
a055b4fe45 | ||
|
|
7305bf772c | ||
|
|
3a4c579c6c | ||
|
|
4ce651e319 | ||
|
|
c3adfe7031 | ||
|
|
b7868e804d | ||
|
|
3ea8239c77 | ||
|
|
ddb7369994 | ||
|
|
97b0260921 | ||
|
|
213f896ab8 | ||
|
|
9ef79e677c | ||
|
|
8e03194a65 | ||
|
|
4c6f9d6490 | ||
|
|
95e14227f0 | ||
|
|
3117046342 | ||
|
|
b68aecb5cc | ||
|
|
1bf5e9c3bc | ||
|
|
1c9b155d97 | ||
|
|
0b52dfd852 | ||
|
|
7cc3a7f9a1 | ||
|
|
74df2dc164 | ||
|
|
b40e6bb412 | ||
|
|
10c9a54896 | ||
|
|
1f4d211c32 | ||
|
|
b736a3aeb8 | ||
|
|
b3050e62b3 | ||
|
|
675fd1ba9f | ||
|
|
b1579b1fbd | ||
|
|
84d4aebdeb | ||
|
|
6b1c0590ce | ||
|
|
b7fd52fa6e | ||
|
|
904dbd6d5e | ||
|
|
c6c693d112 | ||
|
|
3f946f9ada | ||
|
|
35482abf7a | ||
|
|
8b77b3a6df | ||
|
|
8240f56ecd | ||
|
|
77dd600393 | ||
|
|
1149195a23 | ||
|
|
800162ce27 | ||
|
|
9b473a8729 | ||
|
|
c553a09cb4 | ||
|
|
2702257d71 | ||
|
|
d41162a89b | ||
|
|
60999a7533 | ||
|
|
09f20e8149 | ||
|
|
bb631556ba | ||
|
|
32f708a014 | ||
|
|
249cc67140 | ||
|
|
301d3a79b7 | ||
|
|
d5d86afed7 | ||
|
|
6e4ea15d82 | ||
|
|
79aa64c999 | ||
|
|
fa405f4cce | ||
|
|
3fa2cfd860 | ||
|
|
6dad655704 | ||
|
|
2a6406a8fb | ||
|
|
eca3685899 | ||
|
|
afc4bea076 | ||
|
|
ace2568028 | ||
|
|
0382bab537 | ||
|
|
5f535d8bae | ||
|
|
48fe080009 | ||
|
|
e4cab9c2a4 | ||
|
|
61a6b45051 | ||
|
|
03cda9a47a | ||
|
|
f00f679bd2 | ||
|
|
75b1be9864 | ||
|
|
2340efedf5 | ||
|
|
3d0b722409 | ||
|
|
777d31e562 | ||
|
|
47d602302f | ||
|
|
615c3c1cf4 | ||
|
|
87002430f4 | ||
|
|
1bb0bce869 | ||
|
|
4d82231af4 | ||
|
|
9e74a49644 | ||
|
|
29d40f4d54 | ||
|
|
dd0314d235 | ||
|
|
fd3260faff | ||
|
|
c05f0275de | ||
|
|
ed13ab3fdd | ||
|
|
5938ddbb35 | ||
|
|
99672fc645 | ||
|
|
3a73cb900d | ||
|
|
87196e0cd1 | ||
|
|
7c171b3a76 | ||
|
|
3694a27752 | ||
|
|
b5b6b5227d | ||
|
|
5ac365a990 | ||
|
|
2361b16c30 | ||
|
|
023fb03490 | ||
|
|
4a0d104a96 | ||
|
|
f85351b163 | ||
|
|
140ad7cac4 | ||
|
|
d0b4b488b1 | ||
|
|
da69186239 | ||
|
|
2ea6058113 | ||
|
|
8913a5ea87 | ||
|
|
de5ff58d18 | ||
|
|
b658e37fcb | ||
|
|
99efbd153f | ||
|
|
09a0630c6b | ||
|
|
cc7dd80a8a | ||
|
|
a1906fd925 | ||
|
|
f48632d7f7 | ||
|
|
f3d31a3f0f | ||
|
|
7aaec2ea43 | ||
|
|
40b7dbc211 | ||
|
|
1cec999bfd | ||
|
|
b33f598742 | ||
|
|
19b2b55051 | ||
|
|
b6235ec7dd | ||
|
|
108a39472a | ||
|
|
63c97e18ad | ||
|
|
19f46b098a | ||
|
|
03ad5239e9 | ||
|
|
720a076648 | ||
|
|
80eef76604 | ||
|
|
feb9c65733 | ||
|
|
7597112520 | ||
|
|
d54044760d | ||
|
|
b369d2d22a | ||
|
|
2e3db496c0 | ||
|
|
97e2f23093 | ||
|
|
42b79c3e30 | ||
|
|
d2fe84a81e | ||
|
|
c3bb902e67 | ||
|
|
dde98e470b | ||
|
|
822fa9714c | ||
|
|
90659afc36 | ||
|
|
7310cbe636 | ||
|
|
0857007e15 | ||
|
|
41b00cb293 | ||
|
|
89466a0995 | ||
|
|
800315fb7c | ||
|
|
8b037e8bac | ||
|
|
085f7e68ef | ||
|
|
5c075a5a7d | ||
|
|
01bc2b838a | ||
|
|
0d495e979d | ||
|
|
6d66419fa6 | ||
|
|
96f629a1b7 | ||
|
|
35df28ea90 | ||
|
|
ca98a8ee2e | ||
|
|
dc09364348 | ||
|
|
423601d840 | ||
|
|
09ee34f0d3 | ||
|
|
98fef43ece | ||
|
|
41bb35f7c6 | ||
|
|
827339ad0e | ||
|
|
9dbdc559cd | ||
|
|
bce328a440 | ||
|
|
84e9cf1ac3 | ||
|
|
5fd5834c8d | ||
|
|
73a783bb97 | ||
|
|
1a5dd023e8 | ||
|
|
e4e011e183 | ||
|
|
4b594290e9 | ||
|
|
c71a0f2a75 | ||
|
|
fd6ab841bf | ||
|
|
2cf27f08b9 | ||
|
|
97f531bbec | ||
|
|
ac7d5e216e | ||
|
|
ee8b373e77 | ||
|
|
ecd1da5fc4 | ||
|
|
cdcf340a75 | ||
|
|
d8941c14b8 | ||
|
|
bb3ca059b0 | ||
|
|
d994a01a10 | ||
|
|
bb432dff18 | ||
|
|
9ce7311f6d | ||
|
|
a468e04bad | ||
|
|
a5dcc8be52 | ||
|
|
f905ed0065 | ||
|
|
0f80375242 | ||
|
|
ed8c9df4c3 | ||
|
|
7bc9f2f75b | ||
|
|
b0c635463a | ||
|
|
5ccf713ef3 | ||
|
|
16e92e925c | ||
|
|
a2ec7d3ccd | ||
|
|
ae3e072275 | ||
|
|
9e195dd1d9 | ||
|
|
8432299660 | ||
|
|
b9f41570bf | ||
|
|
5ead251ea7 | ||
|
|
fc10a77372 | ||
|
|
41ef8c6e97 | ||
|
|
ef0658de24 | ||
|
|
fedb068221 | ||
|
|
5a66c39313 | ||
|
|
ce6ca288b7 | ||
|
|
65541a7001 | ||
|
|
24a5c6be43 | ||
|
|
c6cf6571b6 | ||
|
|
8ea990d070 | ||
|
|
f4a32170d5 | ||
|
|
073508675e | ||
|
|
020c7ec9fa | ||
|
|
f12707d958 | ||
|
|
82c2e20fce | ||
|
|
11f0ee12cf | ||
|
|
ca64c0b219 | ||
|
|
1d6ca0a93a | ||
|
|
e34c6fc843 | ||
|
|
51e139712d | ||
|
|
9daf001312 | ||
|
|
27353ee0c2 | ||
|
|
b6643d89c0 | ||
|
|
9503586b8a | ||
|
|
317735c69a | ||
|
|
5c26000f90 | ||
|
|
139a47b858 | ||
|
|
2f2e05101b | ||
|
|
36f7d9711d | ||
|
|
4e89c8d53f | ||
|
|
5902475778 | ||
|
|
abdba774cc | ||
|
|
c1d10322cb | ||
|
|
90fdf4982e | ||
|
|
f5e5d431c7 | ||
|
|
9d9fb51a7d | ||
|
|
997b5d8d19 | ||
|
|
ea19dd8e0a | ||
|
|
983663a3fa | ||
|
|
3809519331 | ||
|
|
fe7d20778d | ||
|
|
f6bf8b3193 | ||
|
|
5d5ee3cdaa | ||
|
|
ed4da2ab50 | ||
|
|
9c8ed912bb | ||
|
|
c58551adea | ||
|
|
f44618d915 | ||
|
|
aeb26c748a | ||
|
|
c75f0b1312 | ||
|
|
6138fe2e35 | ||
|
|
8d3f99392f | ||
|
|
2fd4792ed4 | ||
|
|
757a822e34 | ||
|
|
03ffe4a911 | ||
|
|
3b83ecd85d | ||
|
|
80145dc114 | ||
|
|
5612f432a5 | ||
|
|
023fd55a70 | ||
|
|
897f3881b2 | ||
|
|
730aa548d4 | ||
|
|
f1a3d775cc | ||
|
|
71d79ec6ef | ||
|
|
4118a21d1d | ||
|
|
be87eec4e5 | ||
|
|
230433de05 | ||
|
|
e3fdf1cc41 | ||
|
|
f1f4ba5dfa | ||
|
|
d9d5de5102 | ||
|
|
7bf74d8727 | ||
|
|
937bdcf5c7 | ||
|
|
0a86240f90 | ||
|
|
ea3c28791d | ||
|
|
1ff829182b | ||
|
|
23fea1bf99 | ||
|
|
496605c539 | ||
|
|
051784e52b | ||
|
|
dcd6fa977d | ||
|
|
411987fb69 | ||
|
|
ac88b70b1c | ||
|
|
36e053068e | ||
|
|
7944c4a414 | ||
|
|
325c2b45e4 | ||
|
|
477fd5164e | ||
|
|
acfdb04b09 | ||
|
|
50073f0512 | ||
|
|
6ab61215b5 | ||
|
|
b5d4bfd488 | ||
|
|
24297dbead | ||
|
|
ad0c17610e | ||
|
|
a54f6f3a89 | ||
|
|
3212260d8e | ||
|
|
8ec7cd3c3c | ||
|
|
32c09c896c | ||
|
|
71c1d77498 | ||
|
|
06fc05e6b3 | ||
|
|
6b491ef696 | ||
|
|
5e23ae7926 | ||
|
|
f62c6b980e | ||
|
|
d930e09065 | ||
|
|
10a6dae2cf | ||
|
|
c099a16a49 | ||
|
|
198e6a11d2 | ||
|
|
8c66e20961 | ||
|
|
0393128be5 | ||
|
|
69ff6be5f4 | ||
|
|
8685362344 | ||
|
|
7ea68e9a09 | ||
|
|
ebede00841 | ||
|
|
73ce298d6a | ||
|
|
69cd561dc0 | ||
|
|
7af1a55e2a | ||
|
|
c1272d9a9f | ||
|
|
f3e28f7cd4 | ||
|
|
412a7ceba1 | ||
|
|
73bf34c422 | ||
|
|
3d2b139214 | ||
|
|
658f0ff587 | ||
|
|
621893fdcc | ||
|
|
a6e94b4173 | ||
|
|
f9098c3203 | ||
|
|
b1c0218ad9 | ||
|
|
4058843b1a | ||
|
|
36aea652d6 | ||
|
|
7deaf27526 | ||
|
|
fceb5904fe | ||
|
|
fa86225ae3 | ||
|
|
7fe43ca0ff | ||
|
|
6511aca103 | ||
|
|
833d530c2f | ||
|
|
3acd8d179e | ||
|
|
828bb76057 | ||
|
|
d4ab4e3c7b | ||
|
|
0346630f02 | ||
|
|
d74e70893e | ||
|
|
55c6202c5e | ||
|
|
63cb48ed23 | ||
|
|
c2d552362a | ||
|
|
364c8998fa | ||
|
|
8cb4f9ccda | ||
|
|
b46021db49 | ||
|
|
7552fd7637 | ||
|
|
6eaa21bf66 | ||
|
|
a0658d93d8 | ||
|
|
8d87c89da2 | ||
|
|
6621da8639 | ||
|
|
6e8c93657e | ||
|
|
0b5d7c8ca1 | ||
|
|
71c86a637c | ||
|
|
4dc1102629 | ||
|
|
7fdd463201 | ||
|
|
f3616f06e8 | ||
|
|
c1f0fbbfd9 | ||
|
|
3597c52af8 | ||
|
|
38eb45cc81 | ||
|
|
94b23241eb | ||
|
|
a851ccdfe7 | ||
|
|
4b7b3b3507 | ||
|
|
1a42ce90ab | ||
|
|
70a45bdb03 | ||
|
|
7644313bbf | ||
|
|
e067419e00 | ||
|
|
cadf057518 | ||
|
|
4d534dabba | ||
|
|
e60e95c899 | ||
|
|
413e8dcf7f | ||
|
|
a0b6e9c00c | ||
|
|
25e649b4d8 | ||
|
|
1146d9763a | ||
|
|
afa8cbac88 | ||
|
|
2f97fbe65e | ||
|
|
002efbc506 | ||
|
|
f67aa65fa2 | ||
|
|
983cd266f5 | ||
|
|
0fcd2c990b | ||
|
|
0bb8bb9c01 | ||
|
|
c8eead9f82 | ||
|
|
29012dd05f | ||
|
|
73dd61762f | ||
|
|
0fedd033d3 | ||
|
|
fc6ce53b8a | ||
|
|
02a735c53a | ||
|
|
d6208eabd3 | ||
|
|
08faad0f41 | ||
|
|
a7ac4120f9 | ||
|
|
16236dda09 | ||
|
|
0746879dd5 | ||
|
|
dabd26b604 | ||
|
|
d6f0bcdb16 | ||
|
|
e560455ef8 | ||
|
|
fe5111a105 | ||
|
|
9924e5a0d9 | ||
|
|
3131bfa683 | ||
|
|
5841095e08 | ||
|
|
8c64b81a12 | ||
|
|
846c040cd6 | ||
|
|
d316d1bf1f | ||
|
|
d0cd7915b7 | ||
|
|
4cac6970f6 | ||
|
|
f1f9a02816 | ||
|
|
34ad979dd0 | ||
|
|
5e1971bfb8 | ||
|
|
a085b9b10a | ||
|
|
7c81c913cb | ||
|
|
a65436c16b | ||
|
|
40216397a6 | ||
|
|
7608b23131 | ||
|
|
940ff7acfa | ||
|
|
b2fa253fb4 | ||
|
|
1f5670bc39 | ||
|
|
f6125f0e75 | ||
|
|
0929d38d3e | ||
|
|
11cba07198 | ||
|
|
8a36398cb7 | ||
|
|
703f33629a | ||
|
|
c716b20887 | ||
|
|
41b333212d | ||
|
|
932cd5fb36 | ||
|
|
a3b62efa1c | ||
|
|
282e7cce8c | ||
|
|
59fa67a999 | ||
|
|
c297c15139 | ||
|
|
c5434df5e4 | ||
|
|
5e8197cd88 | ||
|
|
651cb6c806 | ||
|
|
1ea005efaf | ||
|
|
adbaefece9 | ||
|
|
1e1533a0c9 | ||
|
|
93431df62b | ||
|
|
76fcef69a4 | ||
|
|
c05437f58c | ||
|
|
90ac325cbb | ||
|
|
26508ae87b | ||
|
|
385764c3be | ||
|
|
fea6155306 | ||
|
|
174e7cb0e3 | ||
|
|
413feb6372 | ||
|
|
494bb8ddcc | ||
|
|
71e73e9302 | ||
|
|
7a5bd01ed5 | ||
|
|
b6d9a20d8f | ||
|
|
7d53d444bd | ||
|
|
a8a726216e | ||
|
|
b6258dd32c | ||
|
|
5a0d0f98d3 | ||
|
|
b2cf5c3442 | ||
|
|
47fe93cd19 | ||
|
|
f8b9b9e664 | ||
|
|
2045351ebd | ||
|
|
265870eff7 | ||
|
|
63e51e0e45 | ||
|
|
858333bc57 | ||
|
|
965c6d4c92 | ||
|
|
e82d444cf3 | ||
|
|
d144cd6809 | ||
|
|
49abdf52c2 | ||
|
|
e5c9e339e9 | ||
|
|
d4debd20c4 | ||
|
|
ddd607e1ca | ||
|
|
4755bbd549 | ||
|
|
5566435964 | ||
|
|
6e29d1a1a7 | ||
|
|
4f7f8e8315 | ||
|
|
9b79f46920 | ||
|
|
722dd4ad66 | ||
|
|
c8f038aad8 | ||
|
|
59336c3418 | ||
|
|
5d028394d1 | ||
|
|
19212545c1 | ||
|
|
199d4abeb7 | ||
|
|
cdd8bd0477 | ||
|
|
3b87a0f0f7 | ||
|
|
2ffbad8593 | ||
|
|
d42caccff8 | ||
|
|
8f4f34dc9f | ||
|
|
c2dfb09ec8 | ||
|
|
89f524b186 | ||
|
|
db6d36390e | ||
|
|
2695aeb5a9 | ||
|
|
c385fd2f15 | ||
|
|
3d7ca884da | ||
|
|
c4b3f3a58a | ||
|
|
2e92cc02ea | ||
|
|
0a230dcea1 | ||
|
|
5bc0de8cff | ||
|
|
f647472325 | ||
|
|
24682d70ce | ||
|
|
91b1ae1766 | ||
|
|
7671c7ec72 | ||
|
|
4847564447 | ||
|
|
ac462e3a8e | ||
|
|
8469df52e2 | ||
|
|
9650c32faf | ||
|
|
129d8badd0 | ||
|
|
02b5df25f0 | ||
|
|
00fb4ffc85 | ||
|
|
44ca4e7620 | ||
|
|
39d213d149 | ||
|
|
bea73cd9c8 | ||
|
|
8fe24a56ef | ||
|
|
723088b122 | ||
|
|
db7d64b9a1 | ||
|
|
45cfece1c0 | ||
|
|
9b5e964c68 | ||
|
|
1b0c89a704 | ||
|
|
2b78e0bae5 | ||
|
|
c661177de8 | ||
|
|
02bf964622 | ||
|
|
86baf16f2d | ||
|
|
548ce2f981 | ||
|
|
f473d47028 | ||
|
|
6e76bcb0f4 | ||
|
|
51101eba54 | ||
|
|
3dd4ccba17 | ||
|
|
c3be0b01ca | ||
|
|
bf9a2179ae | ||
|
|
5e0cfdbfcd | ||
|
|
bfe9747658 | ||
|
|
39ef4b08f8 | ||
|
|
f039d13bd2 | ||
|
|
1f95761d38 | ||
|
|
c3e865b407 | ||
|
|
d4fe97860a | ||
|
|
041801a319 | ||
|
|
1e474b25f3 | ||
|
|
222e230cc2 | ||
|
|
53f48e7483 | ||
|
|
4470dc96b7 | ||
|
|
e998f92045 | ||
|
|
62b88c2a0b | ||
|
|
f865af018f | ||
|
|
84c5a9a019 | ||
|
|
43fae83603 | ||
|
|
065d6f7ba8 | ||
|
|
2e6bb9ab66 | ||
|
|
d2226c0b00 | ||
|
|
b84bcfcfaa | ||
|
|
c1ff1575a4 | ||
|
|
a716cb1357 | ||
|
|
a02697d3aa | ||
|
|
a50b658f28 | ||
|
|
179f74dbd0 | ||
|
|
35f1f1f92c | ||
|
|
f81128227c | ||
|
|
0a7caf71c7 | ||
|
|
2d34c6962b | ||
|
|
8797e8e869 | ||
|
|
f416fdc875 | ||
|
|
90e8c21182 | ||
|
|
a10e922a7c | ||
|
|
49b1a3a29b | ||
|
|
f4c16a5377 | ||
|
|
675c6586d2 | ||
|
|
9570fc1cb6 | ||
|
|
cce5caec06 | ||
|
|
64f521ba00 | ||
|
|
834739a878 | ||
|
|
8b29be3f72 | ||
|
|
4d4e4299ab | ||
|
|
5d8e623e1e | ||
|
|
f8c74c21b9 | ||
|
|
1b6a2938f4 | ||
|
|
5bac6e3936 | ||
|
|
fd18581841 | ||
|
|
e380a9d0b0 | ||
|
|
8e67f5544d | ||
|
|
b2323784ae | ||
|
|
48387ed576 | ||
|
|
d4b101d4b0 | ||
|
|
063c19996b | ||
|
|
bf55efeb56 | ||
|
|
b2fe96600f | ||
|
|
13a1356093 | ||
|
|
51c6f4321c | ||
|
|
49c8b24f6c | ||
|
|
f86af9c8b8 | ||
|
|
b709333b1b | ||
|
|
5cf26dbd86 | ||
|
|
0ba28e14c8 | ||
|
|
2457a88778 | ||
|
|
48aac68248 | ||
|
|
c39b5a5d79 | ||
|
|
de7005a82d | ||
|
|
70f76fd592 | ||
|
|
7c5b9e128c | ||
|
|
de0e824b30 | ||
|
|
870c84d318 | ||
|
|
016846b5b4 | ||
|
|
4af8304798 | ||
|
|
1399f1d197 | ||
|
|
e8487b4786 | ||
|
|
a70b53b6e6 | ||
|
|
00fe260108 | ||
|
|
03a88069db | ||
|
|
6322b27c94 | ||
|
|
ab85118e68 | ||
|
|
bacac276ff | ||
|
|
d0fa87009d | ||
|
|
17d79f4de9 | ||
|
|
7e5fcfe8d5 | ||
|
|
6a57f5bfdf | ||
|
|
0e576754bb | ||
|
|
b54952855b | ||
|
|
15e650ef28 | ||
|
|
d6de7cce24 | ||
|
|
8ecd2cfc6a | ||
|
|
4409c3e1d3 | ||
|
|
ac747bcf05 | ||
|
|
6def65bb0d | ||
|
|
92677c4ad2 | ||
|
|
847d3a331b | ||
|
|
51d30f3275 | ||
|
|
0f0d959118 | ||
|
|
3582ca7c17 | ||
|
|
6fd267a80a | ||
|
|
932a5e2d47 | ||
|
|
1d91d4d065 | ||
|
|
fa42ea7cdf | ||
|
|
9e9dea6e57 | ||
|
|
a18520276a | ||
|
|
667f163853 | ||
|
|
4c49d0b32d | ||
|
|
b7cd39d373 | ||
|
|
4b06a29f10 | ||
|
|
ae4962eb45 | ||
|
|
19b7a36aa1 | ||
|
|
1040cca267 | ||
|
|
58e8145f0e | ||
|
|
5b3dc2bc48 | ||
|
|
8d3a3587d1 | ||
|
|
b0731d4fbd | ||
|
|
4b57c78c8e | ||
|
|
241eef173f | ||
|
|
b0b2954311 | ||
|
|
83c21e2304 | ||
|
|
30369de1f5 | ||
|
|
1f5a8f7694 | ||
|
|
6e060c9038 | ||
|
|
70da1a3a5b | ||
|
|
9d7244131f | ||
|
|
9f0923c955 | ||
|
|
c383c2cc54 | ||
|
|
586f8c2c18 | ||
|
|
40554eb93e | ||
|
|
963fc97f61 | ||
|
|
049bc928ce | ||
|
|
288afeeb1f | ||
|
|
6661583a0f | ||
|
|
8f5df65f0e | ||
|
|
5112af8ba1 | ||
|
|
0f39a1c1fa | ||
|
|
07f23e4d3b | ||
|
|
6f7e1dbf18 | ||
|
|
960f5a6c66 | ||
|
|
0cf1d64e32 | ||
|
|
fda8c36867 | ||
|
|
4bf9d9081e | ||
|
|
54a38cd645 | ||
|
|
66c4d3da91 | ||
|
|
46e79e5a53 | ||
|
|
9436ee048d | ||
|
|
2da9fcdedd | ||
|
|
0b34504abf | ||
|
|
610f056dfd | ||
|
|
49f1ca2329 | ||
|
|
494138719f | ||
|
|
291b15956b | ||
|
|
fa72cd0f7b | ||
|
|
5cd45f3493 | ||
|
|
3befab9964 | ||
|
|
ff219c0675 | ||
|
|
2f79fa2d50 | ||
|
|
ff182a23e8 | ||
|
|
71ac63b0c5 | ||
|
|
8675da453f | ||
|
|
1c2a8690fc | ||
|
|
afe21f9930 | ||
|
|
311b15cd1c | ||
|
|
bbe2346bd0 | ||
|
|
2e7df28b9e | ||
|
|
1401875bca | ||
|
|
de176466df | ||
|
|
6ce2a475a9 | ||
|
|
ac998a6082 | ||
|
|
a6467ee773 | ||
|
|
77fc2dee89 | ||
|
|
ef82831e99 | ||
|
|
7a4ec597dd | ||
|
|
05c2425bc6 | ||
|
|
e7619d01a7 | ||
|
|
be6e1a3449 | ||
|
|
4a003cf617 | ||
|
|
5bcb6eafb1 | ||
|
|
5c0d0cf598 | ||
|
|
46e1fd4d0b | ||
|
|
295198bad0 | ||
|
|
92e5475afb | ||
|
|
026f78f686 | ||
|
|
42b89944df | ||
|
|
7a877b0d9b | ||
|
|
e5189ed623 | ||
|
|
a4f59d6096 | ||
|
|
2470f8cba7 | ||
|
|
9cb97c68ed | ||
|
|
7f8dd45c21 | ||
|
|
39fa5bb997 | ||
|
|
0d69a3ee9a | ||
|
|
f9ea0cd1a1 | ||
|
|
ad51b03296 | ||
|
|
cb6139c736 | ||
|
|
4bae678f69 | ||
|
|
e8ae9e2fe5 | ||
|
|
d3f4c62550 | ||
|
|
06e7e4482c | ||
|
|
3099889cff | ||
|
|
e842246202 | ||
|
|
406d159852 | ||
|
|
e3ad602add | ||
|
|
8019d4457b | ||
|
|
7e4d2ac1d0 | ||
|
|
1401b59b13 | ||
|
|
efdfd0af67 | ||
|
|
3e1af7ef57 | ||
|
|
f6724b2d8c | ||
|
|
ea611a518d | ||
|
|
6b550a2cfa | ||
|
|
fe7997b7b1 | ||
|
|
3ea28ee963 | ||
|
|
f18a7fe6fa | ||
|
|
cece17cbe6 | ||
|
|
d808aca25e | ||
|
|
eb90e036ca | ||
|
|
559fc8b837 | ||
|
|
059aa62ed8 | ||
|
|
6e6dfc6912 | ||
|
|
6200512cae | ||
|
|
171ee84c1b | ||
|
|
5a27662f47 | ||
|
|
a569ffa003 | ||
|
|
1ea52433d5 | ||
|
|
a9f65ddc81 | ||
|
|
222c76bdae | ||
|
|
7d46392174 | ||
|
|
13089e80f8 | ||
|
|
a484fc4e2a | ||
|
|
8f9c186343 | ||
|
|
c9065c88f3 | ||
|
|
52e34c5e75 | ||
|
|
a4acc55994 | ||
|
|
2c2fac2ac1 | ||
|
|
2ded71fa90 | ||
|
|
9a5700b48b | ||
|
|
0bc3a9f0e7 | ||
|
|
2ff292b55f | ||
|
|
1f8ef33c5c | ||
|
|
b37ef2d144 | ||
|
|
b045fb1c8a | ||
|
|
9b3c75f0a6 | ||
|
|
3e171e20c3 | ||
|
|
a6200e07f3 | ||
|
|
a4ec078c17 | ||
|
|
ab5f6abd9e | ||
|
|
b3d41b5fc2 | ||
|
|
ad870fe837 | ||
|
|
88c4d9e56f | ||
|
|
b4414bcaf2 | ||
|
|
3212d90436 | ||
|
|
117b237bda | ||
|
|
0616b0bc7c | ||
|
|
4e04675dfe | ||
|
|
6a5cf5f35d | ||
|
|
d0f9b272f1 | ||
|
|
2e7ffb217e | ||
|
|
31a401b55d | ||
|
|
c16be0a85c | ||
|
|
15cf13d4de | ||
|
|
6257b7fdc1 | ||
|
|
3b17b44de5 | ||
|
|
7a724e76b7 | ||
|
|
8b3399351c | ||
|
|
3d9b034ab8 | ||
|
|
bd1fb83325 | ||
|
|
b20386cd18 | ||
|
|
c4512dc3a9 | ||
|
|
7064a48744 | ||
|
|
4e1ade34b8 | ||
|
|
4a014a338d | ||
|
|
523c31d3ad | ||
|
|
7a627a5bfc | ||
|
|
9864d43e95 | ||
|
|
cf2a16a746 | ||
|
|
25df2096fd | ||
|
|
47f5cd892a | ||
|
|
2aa3451c24 | ||
|
|
f3f679831b | ||
|
|
4bf00fac7c | ||
|
|
e867277a7f | ||
|
|
c9a3fe625c | ||
|
|
7e3db76c9f | ||
|
|
5fef193029 | ||
|
|
2289b43516 | ||
|
|
affca62d1a | ||
|
|
3041dee3ce | ||
|
|
ecde3f6375 | ||
|
|
b2823f6978 | ||
|
|
06b0b748a7 | ||
|
|
6313184711 | ||
|
|
a04c613905 | ||
|
|
26d106ca9e | ||
|
|
6ce6fd6c21 | ||
|
|
5c68d786ce | ||
|
|
ac5ec22d5e | ||
|
|
bb431bf58b | ||
|
|
bb7030d6e7 | ||
|
|
08e5996705 | ||
|
|
01e4edc7d8 | ||
|
|
524e3499a1 | ||
|
|
1b417549d6 | ||
|
|
443617e66e | ||
|
|
df5f1097e6 | ||
|
|
5a8e4027c6 | ||
|
|
77d1acd691 | ||
|
|
615df0e77a | ||
|
|
4496f58d84 | ||
|
|
90d5022933 | ||
|
|
c3a298739b | ||
|
|
2775d32ca0 | ||
|
|
2aa712f72b | ||
|
|
48e64534b4 | ||
|
|
504d0050c8 | ||
|
|
d840d2d38d | ||
|
|
891309ee9f | ||
|
|
e307ec2744 | ||
|
|
1aa325b429 | ||
|
|
5bf3f726e8 | ||
|
|
906ef63a50 | ||
|
|
dbc5656555 | ||
|
|
91a2cc075a | ||
|
|
098eabf308 | ||
|
|
5de8c055c1 | ||
|
|
3acb1b9a10 | ||
|
|
929a394bee | ||
|
|
e835e0a671 | ||
|
|
2b1c44b6f5 | ||
|
|
fbc94ac4f2 | ||
|
|
14a76db7d8 | ||
|
|
f2cec72378 | ||
|
|
2fe14c7257 | ||
|
|
5cebef0f59 | ||
|
|
49eeb7e8a2 | ||
|
|
bba4750ec3 | ||
|
|
8049e0d965 | ||
|
|
45ff13c7ce | ||
|
|
449d4ae18e | ||
|
|
db457d169c | ||
|
|
17efdb8096 | ||
|
|
f13031d120 | ||
|
|
d751a23df1 | ||
|
|
b8b22a6f8e | ||
|
|
b891be3607 | ||
|
|
a4840aa2a6 | ||
|
|
b1c2645ab0 | ||
|
|
3d90ee0b41 | ||
|
|
3da57560e7 | ||
|
|
d4dbd8a535 | ||
|
|
791f173cea | ||
|
|
7b5ab3defe | ||
|
|
a272fbb542 | ||
|
|
d754acf77d | ||
|
|
454ef107ae | ||
|
|
68b1234e81 | ||
|
|
6349ae2c17 | ||
|
|
ca9c467644 | ||
|
|
b2e87d7457 | ||
|
|
ba6750d7d6 | ||
|
|
d6518019dd | ||
|
|
fa77a95d93 | ||
|
|
d66c6b59a2 | ||
|
|
9e3fbbad10 | ||
|
|
9623264e56 | ||
|
|
bef1e8f49a | ||
|
|
47ef7120f7 | ||
|
|
7f0924fb52 | ||
|
|
7312a899e1 | ||
|
|
94955cb0d8 | ||
|
|
2ec9c02d0c | ||
|
|
208c7c0aa8 | ||
|
|
5e321e9a27 | ||
|
|
1dfe004370 | ||
|
|
5aec4f3f44 | ||
|
|
b12aaf003e | ||
|
|
1c80faede4 | ||
|
|
4c8819c039 | ||
|
|
d3c297d24f | ||
|
|
18150eb8ab | ||
|
|
1c00db1978 | ||
|
|
0ea46e9ee2 | ||
|
|
70e9b26c4d | ||
|
|
6af2ee0555 | ||
|
|
b68a3b27fa | ||
|
|
3ee819dd4c | ||
|
|
de29a07c35 | ||
|
|
59bbd4d743 | ||
|
|
790a3573e9 | ||
|
|
023f4fde09 | ||
|
|
e29a4db922 | ||
|
|
df06fbe85e | ||
|
|
7a315a056a | ||
|
|
e51d13a47b | ||
|
|
917c22a9eb | ||
|
|
650b5ded35 | ||
|
|
92c53cbfc1 | ||
|
|
2ce02c54b6 | ||
|
|
2ed6ddb92a | ||
|
|
efdbc001ff | ||
|
|
bbc43b7ec6 | ||
|
|
41d07fd670 | ||
|
|
66a33d0c44 | ||
|
|
179c420e9d | ||
|
|
bd4811c485 | ||
|
|
7ea639c078 | ||
|
|
105230eeb0 | ||
|
|
93ec844121 | ||
|
|
fd84c73d6b | ||
|
|
1af963337b | ||
|
|
c99505f39f | ||
|
|
d886a5e607 | ||
|
|
cc21e63b5b | ||
|
|
d18cd3d254 | ||
|
|
b01133020d | ||
|
|
cc9b4ea2bd | ||
|
|
006477172b | ||
|
|
3b2f4cddea | ||
|
|
01fb9f5411 | ||
|
|
4281e02f95 | ||
|
|
fc0ee3d4f8 | ||
|
|
f79121724d | ||
|
|
f50191d538 | ||
|
|
5b1bc4478c | ||
|
|
bb910aa447 | ||
|
|
f13afff928 | ||
|
|
a003e20e51 | ||
|
|
33d5d725d3 | ||
|
|
8fb0485887 | ||
|
|
4850dce79b | ||
|
|
2097e8a361 | ||
|
|
ad27c4b9fe | ||
|
|
2230177680 | ||
|
|
1279015c28 | ||
|
|
23c66964cd | ||
|
|
a180265585 | ||
|
|
4af602b706 | ||
|
|
d445019131 | ||
|
|
f97105ede0 | ||
|
|
76a63f2be5 | ||
|
|
b5d30129f2 | ||
|
|
26c021fc77 | ||
|
|
672db6380d | ||
|
|
feb428356c | ||
|
|
212c4e5bd7 | ||
|
|
1029d0cfa1 | ||
|
|
e55272180d | ||
|
|
332112e4dd | ||
|
|
5f65099af5 | ||
|
|
ebe85219eb | ||
|
|
a8a1c99312 | ||
|
|
adec7f4020 | ||
|
|
02e744d833 | ||
|
|
e70b5cb2c4 | ||
|
|
30213bc4c0 | ||
|
|
acc919b0e7 | ||
|
|
8e21eb0a97 | ||
|
|
3fa33e482a | ||
|
|
d88ff02fa8 | ||
|
|
d46e49df2a | ||
|
|
608b8b727f | ||
|
|
fd3191ad09 | ||
|
|
0b270fe3bb | ||
|
|
9ad0ee6c24 | ||
|
|
c05a01e46f | ||
|
|
4f68d57666 | ||
|
|
fd05028cff | ||
|
|
a28dd9cf6f | ||
|
|
7e7bcae8e7 | ||
|
|
218ef50ff7 | ||
|
|
18a9d99f08 | ||
|
|
53a763f3c1 | ||
|
|
cfd1b1e550 | ||
|
|
f8b6df9487 | ||
|
|
6d8b45817e | ||
|
|
fc9b44db21 | ||
|
|
100405e080 | ||
|
|
4b84a77a5a | ||
|
|
119b494cc6 | ||
|
|
95945a3b20 | ||
|
|
1b613a82f9 | ||
|
|
c691a93ee5 | ||
|
|
16e7e548b3 | ||
|
|
3704e4d478 | ||
|
|
c16ff3e894 | ||
|
|
47006a2a88 | ||
|
|
952a968d3c | ||
|
|
d7f3274b59 | ||
|
|
cfedd153c8 | ||
|
|
b21d1533de | ||
|
|
5f7237309c | ||
|
|
e30b15ceec | ||
|
|
2d62fc5429 | ||
|
|
cd710f37a5 | ||
|
|
b55d2208ae | ||
|
|
ec8a3cee21 | ||
|
|
1c30a646dc | ||
|
|
24357c486a | ||
|
|
f6d00a5030 | ||
|
|
b52cd1200a | ||
|
|
1819dfb311 | ||
|
|
eeb1aed26a | ||
|
|
b63d6cd748 | ||
|
|
da808ca007 | ||
|
|
aa351f0c2d | ||
|
|
9636cbace0 | ||
|
|
188ff04ec7 | ||
|
|
057bb73405 | ||
|
|
5d2e826bae | ||
|
|
166adbb7ba | ||
|
|
8213aa8a68 | ||
|
|
fc08f54394 | ||
|
|
9ec45a889f | ||
|
|
ae6c7d33a8 | ||
|
|
21b61f22f7 | ||
|
|
9516fb4963 | ||
|
|
a5a3a47a69 | ||
|
|
8b38d45883 | ||
|
|
5d8a9251e5 | ||
|
|
84fb2da1d2 | ||
|
|
46d61151f2 | ||
|
|
bd84b44175 | ||
|
|
253673fc9a | ||
|
|
7f6f9ec285 | ||
|
|
88d108f85c | ||
|
|
a9925bcf28 | ||
|
|
f601a45700 | ||
|
|
283d321307 | ||
|
|
ef27cd7862 | ||
|
|
4c1be1aa68 | ||
|
|
eb550aeec3 | ||
|
|
14c4d4fe00 | ||
|
|
eec62c12ff | ||
|
|
b84e191d31 | ||
|
|
ac7994b631 | ||
|
|
ab345ee4ef | ||
|
|
beced802f0 | ||
|
|
b4331b7758 | ||
|
|
ae112ab7d9 | ||
|
|
84985b85e0 | ||
|
|
78d4867f25 | ||
|
|
dbf44cdfd8 | ||
|
|
73d03fecaf | ||
|
|
392ab31b1f | ||
|
|
f05c773bc7 | ||
|
|
3b8ae66db5 | ||
|
|
03ee1b0513 | ||
|
|
f013194265 | ||
|
|
9322c4e7ae | ||
|
|
aafc6af69c | ||
|
|
fbcb522f6c | ||
|
|
4f628e9b33 | ||
|
|
3366540521 | ||
|
|
262f77da9c | ||
|
|
cde1c8bf79 | ||
|
|
c39f0a0aba | ||
|
|
476cfb1f9c | ||
|
|
e6915c669b | ||
|
|
b2987e1d4f | ||
|
|
5eb6a4307d | ||
|
|
a26b42fef8 | ||
|
|
24c8661453 | ||
|
|
4f2a8149fe | ||
|
|
aba8a3cc55 | ||
|
|
7f6edc2749 | ||
|
|
7f28629748 | ||
|
|
2ac9b885c9 | ||
|
|
6d756eab5d | ||
|
|
8649bb86e9 | ||
|
|
7b0e41d478 | ||
|
|
3127afbbc2 | ||
|
|
0d993c0b90 | ||
|
|
424f688406 | ||
|
|
f2f0661ed3 | ||
|
|
14fd1a06fd | ||
|
|
4c94694e78 | ||
|
|
f1b5a26f2d | ||
|
|
357daf2756 | ||
|
|
97f708b9ef | ||
|
|
91f6fe3965 | ||
|
|
85abb5937c | ||
|
|
c03b08e8f2 | ||
|
|
f5cc891cc4 | ||
|
|
89de63edcd | ||
|
|
8551426619 | ||
|
|
d8deab799e | ||
|
|
f4b09a76b8 | ||
|
|
32ec05e04c | ||
|
|
ca6e20573f | ||
|
|
ae7e8df986 | ||
|
|
dc90a13671 | ||
|
|
f294cb4e9a | ||
|
|
ba5a19582e | ||
|
|
da4cc3de20 | ||
|
|
7b6ccf3f30 | ||
|
|
13122b07d4 | ||
|
|
4effc3c3b4 | ||
|
|
fcc74f0ca3 | ||
|
|
8950d57ff8 | ||
|
|
6398c54b49 | ||
|
|
0da88c6a61 | ||
|
|
e62bffd20a | ||
|
|
537cb70a89 | ||
|
|
173bbe52bd | ||
|
|
b924fdc308 | ||
|
|
84a04ecfca | ||
|
|
3935643e35 | ||
|
|
91df9c8f1e | ||
|
|
e8db62f900 | ||
|
|
239d06bbcc | ||
|
|
47a423284b | ||
|
|
2669c8fff0 | ||
|
|
ebaf97a495 | ||
|
|
4afb282ca2 | ||
|
|
25f4f15e76 | ||
|
|
60286f69ad | ||
|
|
addd22ec6c | ||
|
|
0bab09e48e | ||
|
|
cf234ca067 | ||
|
|
a908cc61c9 | ||
|
|
b19c79f48c | ||
|
|
14f0fe1160 | ||
|
|
21c8fd6e98 | ||
|
|
803ff92976 | ||
|
|
5d74a975df | ||
|
|
25802db0d8 | ||
|
|
537a3d3ce8 | ||
|
|
a0e124717b | ||
|
|
320ba89aea | ||
|
|
8514444508 | ||
|
|
b194c0100f | ||
|
|
d7dedf15f1 | ||
|
|
c491fb6c40 | ||
|
|
0e3af4df5d | ||
|
|
32188dcef4 | ||
|
|
0a199a2588 | ||
|
|
8f1893f665 | ||
|
|
4f765fe527 | ||
|
|
7eeb917e16 | ||
|
|
93d8d0f5bf | ||
|
|
6db7cb4a6e | ||
|
|
602ea51bbf | ||
|
|
ec3eab6e72 | ||
|
|
a8bb7811ed | ||
|
|
9ef9e656e2 | ||
|
|
c4bc4a1d9b | ||
|
|
f50445032f | ||
|
|
af00517102 | ||
|
|
9a3c251867 | ||
|
|
daa662ac27 | ||
|
|
0157c42c52 | ||
|
|
ccfb7356eb | ||
|
|
7fa7096d91 | ||
|
|
c5972f1b59 | ||
|
|
c2d97b58ee | ||
|
|
c97b3ffbdf | ||
|
|
3ec02a299c | ||
|
|
2cffb5c952 | ||
|
|
03ca2d614f | ||
|
|
17bf2b0611 | ||
|
|
8c178d92de | ||
|
|
c0a3251f44 | ||
|
|
bfdc6b4504 | ||
|
|
bfdf09908e | ||
|
|
80afe04aa9 | ||
|
|
a9e38b2be6 | ||
|
|
22738982cb | ||
|
|
757c6ef44c | ||
|
|
bf6025c971 | ||
|
|
4252fb6533 | ||
|
|
e451930e7b | ||
|
|
09d6f7b82b | ||
|
|
0f9bcda6ca | ||
|
|
cd367fb0d5 | ||
|
|
cd7010dccb | ||
|
|
3136319352 | ||
|
|
eb00b5f1ca | ||
|
|
b4377e0746 | ||
|
|
5f1d67ee21 | ||
|
|
e932cb2456 | ||
|
|
85abefff74 | ||
|
|
ee4a9a967b | ||
|
|
a3d3dcc44c | ||
|
|
75fbfab95d | ||
|
|
d48a61c597 | ||
|
|
01919a496e | ||
|
|
1bfe5eed15 | ||
|
|
d311884ec0 | ||
|
|
1291b693f0 | ||
|
|
df15fd497d | ||
|
|
23ec8e4687 | ||
|
|
2f7e3bbc85 | ||
|
|
690757b225 | ||
|
|
d3e3561407 | ||
|
|
2e7c48a999 | ||
|
|
76d9a69220 | ||
|
|
6388eed6e3 | ||
|
|
90d737773d | ||
|
|
4f23c40598 | ||
|
|
7bb16515b1 | ||
|
|
36bb2cf62f | ||
|
|
d7f7650e3a | ||
|
|
348847e196 | ||
|
|
95b2650a8a | ||
|
|
14935cb2f7 | ||
|
|
f059640602 | ||
|
|
22fe547215 | ||
|
|
2612134273 | ||
|
|
9630d8200c | ||
|
|
0a5bfa5a55 | ||
|
|
29e51e92a4 | ||
|
|
bd935e9dc2 | ||
|
|
2edee985fa | ||
|
|
7afa58319f | ||
|
|
7d3d4ab73c |
@@ -4,11 +4,17 @@ root = true
|
||||
[*]
|
||||
charset = utf-8
|
||||
indent_style = space
|
||||
indent_size = 2
|
||||
end_of_line = lf
|
||||
insert_final_newline = true
|
||||
trim_trailing_whitespace = true
|
||||
|
||||
|
||||
[*.{js,jsx,ts,tsx,vue}]
|
||||
indent_size = 2
|
||||
|
||||
[*.py]
|
||||
indent_size = 4
|
||||
|
||||
[*.md]
|
||||
insert_final_newline = false
|
||||
trim_trailing_whitespace = false
|
||||
|
||||
32
.github/workflows/jms-build-test.yml
vendored
Normal file
32
.github/workflows/jms-build-test.yml
vendored
Normal file
@@ -0,0 +1,32 @@
|
||||
name: "Run Build Test"
|
||||
on:
|
||||
push:
|
||||
branches:
|
||||
- pr@*
|
||||
- repr@*
|
||||
|
||||
jobs:
|
||||
build:
|
||||
runs-on: ubuntu-latest
|
||||
steps:
|
||||
- uses: actions/checkout@v3
|
||||
|
||||
- uses: docker/setup-qemu-action@v2
|
||||
|
||||
- uses: docker/setup-buildx-action@v2
|
||||
|
||||
- uses: docker/build-push-action@v3
|
||||
with:
|
||||
context: .
|
||||
push: false
|
||||
tags: jumpserver/lina:test
|
||||
file: Dockerfile
|
||||
cache-from: type=gha
|
||||
cache-to: type=gha,mode=max
|
||||
|
||||
- uses: LouisBrunner/checks-action@v1.5.0
|
||||
if: always()
|
||||
with:
|
||||
token: ${{ secrets.GITHUB_TOKEN }}
|
||||
name: Check Build
|
||||
conclusion: ${{ job.status }}
|
||||
3
.github/workflows/release-drafter.yml
vendored
3
.github/workflows/release-drafter.yml
vendored
@@ -31,6 +31,9 @@ jobs:
|
||||
config-name: release-config.yml
|
||||
version: ${{ steps.get_version.outputs.TAG }}
|
||||
tag: ${{ steps.get_version.outputs.TAG }}
|
||||
- uses: actions/setup-node@v2
|
||||
with:
|
||||
node-version: '14.16'
|
||||
|
||||
build-and-release:
|
||||
needs: create-realese
|
||||
|
||||
1
.gitignore
vendored
1
.gitignore
vendored
@@ -16,3 +16,4 @@ tests/**/coverage/
|
||||
*.njsproj
|
||||
*.sln
|
||||
.env.development
|
||||
.python-version
|
||||
|
||||
12
Dockerfile
12
Dockerfile
@@ -1,26 +1,24 @@
|
||||
FROM node:14.16 as stage-build
|
||||
ARG TARGETARCH
|
||||
ARG NPM_REGISTRY="https://registry.npmmirror.com"
|
||||
ENV NPM_REGISTY=$NPM_REGISTRY
|
||||
|
||||
WORKDIR /data
|
||||
|
||||
RUN set -ex \
|
||||
&& npm config set registry ${NPM_REGISTRY} \
|
||||
&& yarn config set registry ${NPM_REGISTRY} \
|
||||
&& yarn config set cache-folder /root/.cache/yarn/lina
|
||||
&& yarn config set registry ${NPM_REGISTRY}
|
||||
|
||||
ADD package.json yarn.lock /data
|
||||
RUN --mount=type=cache,target=/root/.cache/yarn \
|
||||
RUN --mount=type=cache,target=/usr/local/share/.cache/yarn,sharing=locked,id=lina \
|
||||
yarn install
|
||||
|
||||
ARG VERSION
|
||||
ENV VERSION=$VERSION
|
||||
ADD . /data
|
||||
RUN --mount=type=cache,target=/root/.cache/yarn \
|
||||
sed -i "s@Version <strong>.*</strong>@Version <strong>${VERSION}</strong>@g" src/layout/components/Footer/index.vue \
|
||||
RUN --mount=type=cache,target=/usr/local/share/.cache/yarn,sharing=locked,id=lina \
|
||||
sed -i "s@version-dev@${VERSION}@g" src/layout/components/NavHeader/About.vue \
|
||||
&& yarn build
|
||||
|
||||
FROM nginx:alpine
|
||||
FROM nginx:1.24
|
||||
COPY --from=stage-build /data/lina /opt/lina
|
||||
COPY nginx.conf /etc/nginx/conf.d/default.conf
|
||||
|
||||
10
package.json
10
package.json
@@ -12,13 +12,16 @@
|
||||
"build:stage": "vue-cli-service build --mode staging",
|
||||
"preview": "node build/index.js --preview",
|
||||
"lint": "eslint --ext .js,.vue src",
|
||||
"fix": "eslint --ext .js,.vue --fix src",
|
||||
"test:unit": "jest --clearCache && vue-cli-service test:unit",
|
||||
"test:ci": "npm run lint && npm run test:unit",
|
||||
"svgo": "svgo -f src/icons/svg --config=src/icas/svgo.yml",
|
||||
"vue-i18n-extract": "vue-i18n-extract",
|
||||
"vue-i18n-report": "vue-i18n-extract report -v './src/**/*.?(js|vue)' -l './src/i18n/langs/**/*.json'",
|
||||
"vue-i18n-report-json": "vue-i18n-extract report -v './src/**/*.?(js|vue)' -l './src/i18n/langs/**/*.json' -o /tmp/abc.json",
|
||||
"vue-i18n-report-add-miss": "vue-i18n-extract report -v './src/**/*.?(js|vue)' -l './src/i18n/langs/**/*.json' -a"
|
||||
"vue-i18n-report-add-miss": "vue-i18n-extract report -v './src/**/*.?(js|vue)' -l './src/i18n/langs/**/*.json' -a",
|
||||
"diff-i18n": "python ./src/i18n/langs/i18n-util.py diff en ja",
|
||||
"apply-i18n": "python ./src/i18n/langs/i18n-util.py apply en ja"
|
||||
},
|
||||
"dependencies": {
|
||||
"@babel/plugin-proposal-optional-chaining": "^7.13.12",
|
||||
@@ -58,17 +61,19 @@
|
||||
"nprogress": "0.2.0",
|
||||
"path-to-regexp": "2.4.0",
|
||||
"vue": "2.6.10",
|
||||
"vue-codemirror-lite": "^1.0.4",
|
||||
"vue-codemirror": "4.0.6",
|
||||
"vue-cookie": "^1.1.4",
|
||||
"vue-echarts": "^5.0.0-beta.0",
|
||||
"vue-i18n": "^8.15.5",
|
||||
"vue-json-editor": "^1.4.3",
|
||||
"vue-markdown": "^2.2.4",
|
||||
"vue-moment": "^4.1.0",
|
||||
"vue-password-strength-meter": "^1.7.2",
|
||||
"vue-router": "3.0.6",
|
||||
"vue-select": "^3.9.5",
|
||||
"vuejs-logger": "^1.5.4",
|
||||
"vuex": "3.1.0",
|
||||
"xss": "^1.0.14",
|
||||
"xterm": "^4.5.0",
|
||||
"xterm-addon-fit": "^0.3.0",
|
||||
"zxcvbn": "^4.4.2"
|
||||
@@ -92,6 +97,7 @@
|
||||
"eslint": "^5.15.3",
|
||||
"eslint-plugin-vue": "5.2.2",
|
||||
"eslint-plugin-vue-i18n": "^0.3.0",
|
||||
"github-markdown-css": "^5.1.0",
|
||||
"html-webpack-plugin": "3.2.0",
|
||||
"husky": "^4.2.3",
|
||||
"less-loader": "^5.0.0",
|
||||
|
||||
@@ -17,9 +17,15 @@
|
||||
</noscript>
|
||||
<script>
|
||||
window.onload = function() {
|
||||
const baseUrl = "/ui/"
|
||||
if (location.pathname === '/' && baseUrl !== '/') {
|
||||
location.pathname = baseUrl
|
||||
if (location.pathname === '/') {
|
||||
location.pathname = '/ui/'
|
||||
}
|
||||
const pathname = window.location.pathname
|
||||
if (pathname.startsWith('/core')) {
|
||||
return
|
||||
}
|
||||
if(pathname.indexOf('/ui') === -1) {
|
||||
window.location.href = window.location.origin + '/ui/#' + pathname
|
||||
}
|
||||
}
|
||||
</script>
|
||||
|
||||
@@ -194,17 +194,16 @@ td .el-button.el-button--mini {
|
||||
line-height: 34px;
|
||||
}
|
||||
|
||||
.el-select-dropdown.is-multiple .el-select-dropdown__item.selected {
|
||||
color: #606266;
|
||||
background-color: #ddd;
|
||||
font-weight: 400;
|
||||
}
|
||||
|
||||
.el-select-dropdown__item.hover {
|
||||
.option-group .el-select-dropdown__item.hover, .option-group .el-select-dropdown__item.selected {
|
||||
background-color: primary;
|
||||
color: white;
|
||||
}
|
||||
|
||||
.option-group:has(.hover) .el-select-dropdown__item.selected {
|
||||
background-color: light-2;
|
||||
}
|
||||
|
||||
|
||||
.el-select-dropdown__item.is-disabled:hover{
|
||||
color:#c0c4cc;
|
||||
}
|
||||
@@ -215,7 +214,7 @@ td .el-button.el-button--mini {
|
||||
|
||||
.el-select-dropdown.is-multiple .el-select-dropdown__item.selected.hover {
|
||||
color: white;
|
||||
background-color: primary;
|
||||
background-color: light-4;
|
||||
}
|
||||
|
||||
.el-tag.el-tag--info {
|
||||
@@ -331,13 +330,12 @@ td .el-button.el-button--mini {
|
||||
}
|
||||
|
||||
.el-table .cell,
|
||||
.el-table--border td:first-child .cell,
|
||||
.el-table--border th:first-child .cell {
|
||||
.el-table td:first-child .cell,
|
||||
.el-table th:first-child .cell {
|
||||
padding-left: 10px;
|
||||
padding-right: 14px;
|
||||
}
|
||||
|
||||
|
||||
.el-tag--default.el-tag--dark {
|
||||
background-color: #d1dade;
|
||||
color: #5e5e5e;
|
||||
@@ -453,11 +451,56 @@ td .el-button.el-button--mini {
|
||||
|
||||
.el-select-dropdown__item.selected {
|
||||
font-weight: 400;
|
||||
color: #606266;
|
||||
background-color: #ddd;
|
||||
color: white;
|
||||
background-color: primary;
|
||||
}
|
||||
|
||||
.el-input-group__prepend div.el-select .el-input__inner,
|
||||
.el-input-group__prepend div.el-select .el-input__inner:hover {
|
||||
color: #303133;
|
||||
}
|
||||
|
||||
.el-input-group__append, .el-input-group__prepend {
|
||||
color: primary
|
||||
}
|
||||
|
||||
.el-input.is-disabled .el-input__inner {
|
||||
color: $--color-text-primary;
|
||||
cursor: not-allowed;
|
||||
}
|
||||
|
||||
.el-step__description.is-finish {
|
||||
color: #676a6c;
|
||||
}
|
||||
|
||||
.el-alert {
|
||||
border: solid 1px #e7eaec;
|
||||
}
|
||||
|
||||
.el-alert.el-alert--success.is-light {
|
||||
border-color: var(--color-success-light);
|
||||
}
|
||||
|
||||
.el-alert.el-alert--primary.is-light {
|
||||
border-color: var(--color-primary-light);
|
||||
}
|
||||
|
||||
.el-alert.el-alert--info.is-light {
|
||||
border-color: var(--color-info-light);
|
||||
}
|
||||
|
||||
.el-alert.el-alert--warning.is-light {
|
||||
border-color: var(--color-warning-light);
|
||||
}
|
||||
|
||||
.el-alert.el-alert--error.is-light {
|
||||
border-color: var(--color-danger-light);
|
||||
}
|
||||
|
||||
#nprogress .bar {
|
||||
background: light-5!important;
|
||||
}
|
||||
|
||||
#nprogress .peg {
|
||||
box-shadow: 0 0 10px light-5, 0 0 5px light-5!important;
|
||||
}
|
||||
|
||||
File diff suppressed because one or more lines are too long
11
src/App.vue
11
src/App.vue
@@ -1,12 +1,19 @@
|
||||
<template>
|
||||
<div id="app">
|
||||
<router-view />
|
||||
<router-view v-if="isRouterAlive" />
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<script>
|
||||
import { mapState } from 'vuex'
|
||||
|
||||
export default {
|
||||
name: 'App'
|
||||
name: 'App',
|
||||
computed: {
|
||||
...mapState({
|
||||
isRouterAlive: state => state.common.isRouterAlive
|
||||
})
|
||||
}
|
||||
}
|
||||
</script>
|
||||
|
||||
|
||||
@@ -40,3 +40,10 @@ export function getCommandFilterList(data) {
|
||||
})
|
||||
}
|
||||
|
||||
export function getCategoryTypes() {
|
||||
return request({
|
||||
url: '/api/v1/assets/categories/',
|
||||
method: 'get'
|
||||
})
|
||||
}
|
||||
|
||||
|
||||
@@ -1,6 +1,13 @@
|
||||
import request from '@/utils/request'
|
||||
|
||||
export function createSourceIdCache(ids) {
|
||||
ids = ids.map(item => {
|
||||
if (typeof item === 'object' && item.id) {
|
||||
return item.id
|
||||
} else {
|
||||
return item
|
||||
}
|
||||
})
|
||||
return request({
|
||||
url: '/api/v1/common/resources/cache/',
|
||||
method: 'post',
|
||||
|
||||
@@ -1,12 +1,5 @@
|
||||
import request from '@/utils/request'
|
||||
|
||||
export function getTaskDetail(id) {
|
||||
return request({
|
||||
url: `/api/v1/ops/tasks/${id}/`,
|
||||
method: 'get'
|
||||
})
|
||||
}
|
||||
|
||||
export function getAdhocDetail(id) {
|
||||
return request({
|
||||
url: `/api/v1/ops/adhoc/${id}/`,
|
||||
@@ -20,3 +13,34 @@ export function getHistoryExecutionDetail(id) {
|
||||
method: 'get'
|
||||
})
|
||||
}
|
||||
|
||||
export function getTaskDetail(id) {
|
||||
return request({
|
||||
url: `/api/v1/ops/job-execution/task-detail/${id}/`,
|
||||
method: 'get'
|
||||
})
|
||||
}
|
||||
|
||||
export function getJob(id) {
|
||||
return request({
|
||||
url: `/api/v1/ops/jobs/${id}/`,
|
||||
method: 'get'
|
||||
})
|
||||
}
|
||||
|
||||
export function uploadPlaybook(form) {
|
||||
return request({
|
||||
url: '/api/v1/ops/playbooks/',
|
||||
method: 'post',
|
||||
headers: { 'Content-Type': 'multipart/form-data' },
|
||||
data: form
|
||||
})
|
||||
}
|
||||
|
||||
export function renameFile(playbookId, node) {
|
||||
return request({
|
||||
url: `/api/v1/ops/playbook/${playbookId}/file/`,
|
||||
method: 'patch',
|
||||
data: node
|
||||
})
|
||||
}
|
||||
|
||||
@@ -1,43 +0,0 @@
|
||||
import request from '@/utils/request'
|
||||
|
||||
export function getAssetPermissionDetail(id) {
|
||||
return request({
|
||||
url: `/api/v1/perms/asset-permissions/${id}/`,
|
||||
method: 'get'
|
||||
})
|
||||
}
|
||||
|
||||
export function getRemoteAppPermissionDetail(id) {
|
||||
return request({
|
||||
url: `/api/v1/perms/remote-app-permissions/${id}/`,
|
||||
method: 'get'
|
||||
})
|
||||
}
|
||||
|
||||
export function getDatabaseAppPermissionDetail(id) {
|
||||
return request({
|
||||
url: `/api/v1/perms/database-app-permissions/${id}/`,
|
||||
method: 'get'
|
||||
})
|
||||
}
|
||||
|
||||
export function getUserAssetGrantedSystemUsers(userId, assetId) {
|
||||
return request({
|
||||
url: `/api/v1/perms/users/${userId}/assets/${assetId}/system-users/?cache_policy=1`,
|
||||
method: 'get'
|
||||
})
|
||||
}
|
||||
|
||||
export function getMyAssetGrantedSystemUsers(userId, assetId) {
|
||||
return request({
|
||||
url: `/api/v1/perms/users/assets/${assetId}/system-users/?cache_policy=1`,
|
||||
method: 'get'
|
||||
})
|
||||
}
|
||||
|
||||
export function getUserGroupAssetGrantedSystemUsers(gId, assetId) {
|
||||
return request({
|
||||
url: `/api/v1/perms/user-groups/${gId}/assets/${assetId}/system-users/?cache_policy=1`,
|
||||
method: 'get'
|
||||
})
|
||||
}
|
||||
|
||||
@@ -25,12 +25,28 @@ export function importLicense(formData) {
|
||||
data: formData
|
||||
})
|
||||
}
|
||||
export function testLdapSetting(data) {
|
||||
return request({
|
||||
disableFlashErrorMsg: true,
|
||||
url: '/api/v1/settings/ldap/testing/config/',
|
||||
method: 'post',
|
||||
data: data
|
||||
export function testLdapSetting(data, refresh = true) {
|
||||
let url = '/api/v1/settings/ldap/testing/config/'
|
||||
if (refresh) {
|
||||
url = url + '?refresh=1'
|
||||
}
|
||||
return new Promise((resolve, reject) => {
|
||||
request({
|
||||
disableFlashErrorMsg: true,
|
||||
url: url,
|
||||
method: 'post',
|
||||
data: data
|
||||
}).then(res => {
|
||||
if (res.status !== 'running') {
|
||||
resolve(res)
|
||||
} else {
|
||||
setTimeout(() => {
|
||||
resolve(testLdapSetting(data, false))
|
||||
}, 1000)
|
||||
}
|
||||
}).catch(error => {
|
||||
reject(error)
|
||||
})
|
||||
})
|
||||
}
|
||||
|
||||
|
||||
Binary file not shown.
|
Before Width: | Height: | Size: 7.1 KiB |
BIN
src/assets/img/avatar.png
Normal file
BIN
src/assets/img/avatar.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 5.6 KiB |
BIN
src/assets/img/logo_text_green.png
Normal file
BIN
src/assets/img/logo_text_green.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 3.6 KiB |
321
src/components/AccountCreateUpdateForm/index.vue
Normal file
321
src/components/AccountCreateUpdateForm/index.vue
Normal file
@@ -0,0 +1,321 @@
|
||||
<template>
|
||||
<AutoDataForm
|
||||
v-if="!loading"
|
||||
ref="AutoDataForm"
|
||||
v-bind="$data"
|
||||
@submit="confirm"
|
||||
/>
|
||||
</template>
|
||||
|
||||
<script>
|
||||
import AutoDataForm from '@/components/AutoDataForm'
|
||||
import { UpdateToken, UploadSecret } from '@/components/FormFields'
|
||||
import Select2 from '@/components/FormFields/Select2'
|
||||
import AssetSelect from '@/components/AssetSelect'
|
||||
import { encryptPassword } from '@/utils/crypto'
|
||||
import { Required, RequiredChange } from '@/components/DataForm/rules'
|
||||
import AutomationParamsForm from '@/views/assets/Platform/AutomationParamsSetting.vue'
|
||||
|
||||
export default {
|
||||
name: 'AccountCreateForm',
|
||||
components: {
|
||||
AutoDataForm
|
||||
},
|
||||
props: {
|
||||
asset: {
|
||||
type: Object,
|
||||
default: null
|
||||
},
|
||||
platform: {
|
||||
type: Object,
|
||||
default: null
|
||||
},
|
||||
account: {
|
||||
type: Object,
|
||||
default: () => ({})
|
||||
},
|
||||
// 默认组件密码加密
|
||||
encryptPassword: {
|
||||
type: Boolean,
|
||||
default: true
|
||||
},
|
||||
addTemplate: {
|
||||
type: Boolean,
|
||||
default: false
|
||||
}
|
||||
},
|
||||
data() {
|
||||
return {
|
||||
loading: true,
|
||||
usernameChanged: false,
|
||||
defaultPrivilegedAccounts: ['root', 'administrator'],
|
||||
iPlatform: {
|
||||
automation: {},
|
||||
su_enabled: false,
|
||||
protocols: [
|
||||
{
|
||||
name: 'ssh',
|
||||
secret_types: ['password', 'ssh_key', 'token', 'access_key', 'api_key']
|
||||
}
|
||||
]
|
||||
},
|
||||
url: '/api/v1/accounts/accounts/',
|
||||
form: Object.assign({ 'on_invalid': 'error' }, this.account || {}),
|
||||
encryptedFields: ['secret'],
|
||||
fields: [
|
||||
[this.$t('assets.Asset'), ['assets']],
|
||||
[this.$t('accounts.AccountTemplate'), ['template']],
|
||||
[this.$t('common.Basic'), ['name', 'username', 'privileged', 'su_from', 'su_from_username']],
|
||||
[this.$t('assets.Secret'), [
|
||||
'secret_type', 'password', 'ssh_key', 'token',
|
||||
'access_key', 'passphrase', 'api_key'
|
||||
]],
|
||||
[this.$t('common.Other'), ['push_now', 'params', 'on_invalid', 'is_active', 'comment']]
|
||||
],
|
||||
fieldsMeta: {
|
||||
assets: {
|
||||
rules: [Required],
|
||||
component: AssetSelect,
|
||||
label: this.$t('assets.Asset'),
|
||||
el: {
|
||||
multiple: false
|
||||
},
|
||||
hidden: () => {
|
||||
return this.platform || this.asset
|
||||
}
|
||||
},
|
||||
template: {
|
||||
component: Select2,
|
||||
rules: [Required],
|
||||
el: {
|
||||
multiple: false,
|
||||
ajax: {
|
||||
url: '/api/v1/accounts/account-templates/',
|
||||
transformOption: (item) => {
|
||||
return { label: item.name, value: item.id }
|
||||
}
|
||||
}
|
||||
},
|
||||
hidden: () => {
|
||||
return this.platform || this.asset || !this.addTemplate
|
||||
}
|
||||
},
|
||||
on_invalid: {
|
||||
rules: [Required],
|
||||
label: this.$t('accounts.AccountPolicy'),
|
||||
helpText: this.$t('accounts.BulkCreateStrategy'),
|
||||
hidden: () => {
|
||||
return this.platform || this.asset
|
||||
}
|
||||
},
|
||||
name: {
|
||||
rules: [RequiredChange],
|
||||
on: {
|
||||
input: ([value], updateForm) => {
|
||||
if (!this.usernameChanged) {
|
||||
if (!this.account?.name) {
|
||||
updateForm({ username: value })
|
||||
}
|
||||
const maybePrivileged = this.defaultPrivilegedAccounts.includes(value)
|
||||
if (maybePrivileged) {
|
||||
updateForm({ privileged: true })
|
||||
}
|
||||
}
|
||||
}
|
||||
},
|
||||
hidden: () => {
|
||||
return this.addTemplate
|
||||
}
|
||||
},
|
||||
username: {
|
||||
el: {
|
||||
disabled: !!this.account?.name
|
||||
},
|
||||
on: {
|
||||
input: ([value], updateForm) => {
|
||||
this.usernameChanged = true
|
||||
},
|
||||
change: ([value], updateForm) => {
|
||||
const maybePrivileged = this.defaultPrivilegedAccounts.includes(value)
|
||||
if (maybePrivileged) {
|
||||
updateForm({ privileged: true })
|
||||
}
|
||||
}
|
||||
},
|
||||
hidden: () => {
|
||||
return this.addTemplate
|
||||
}
|
||||
},
|
||||
privileged: {
|
||||
hidden: () => {
|
||||
return this.addTemplate
|
||||
}
|
||||
},
|
||||
su_from: {
|
||||
component: Select2,
|
||||
hidden: (formValue) => {
|
||||
return !this.asset?.id || !this.iPlatform.su_enabled
|
||||
},
|
||||
el: {
|
||||
multiple: false,
|
||||
clearable: true,
|
||||
ajax: {
|
||||
url: `/api/v1/accounts/accounts/su-from-accounts/?account=${this.account?.id || ''}&asset=${this.asset?.id || ''}`,
|
||||
transformOption: (item) => {
|
||||
return { label: `${item.name}(${item.username})`, value: item.id }
|
||||
}
|
||||
}
|
||||
}
|
||||
},
|
||||
su_from_username: {
|
||||
label: this.$t('assets.UserSwitchFrom'),
|
||||
hidden: (formValue) => {
|
||||
return this.platform || this.asset || this.addTemplate
|
||||
}
|
||||
},
|
||||
password: {
|
||||
label: this.$t('assets.Password'),
|
||||
component: UpdateToken,
|
||||
hidden: (formValue) => formValue.secret_type !== 'password' || this.addTemplate
|
||||
},
|
||||
ssh_key: {
|
||||
label: this.$t('assets.PrivateKey'),
|
||||
component: UploadSecret,
|
||||
hidden: (formValue) => formValue.secret_type !== 'ssh_key' || this.addTemplate
|
||||
},
|
||||
passphrase: {
|
||||
label: this.$t('assets.Passphrase'),
|
||||
component: UpdateToken,
|
||||
hidden: (formValue) => formValue.secret_type !== 'ssh_key' || this.addTemplate
|
||||
},
|
||||
token: {
|
||||
label: this.$t('assets.Token'),
|
||||
component: UploadSecret,
|
||||
hidden: (formValue) => formValue.secret_type !== 'token' || this.addTemplate
|
||||
},
|
||||
access_key: {
|
||||
id: 'access_key',
|
||||
label: this.$t('assets.AccessKey'),
|
||||
component: UploadSecret,
|
||||
hidden: (formValue) => formValue.secret_type !== 'access_key' || this.addTemplate
|
||||
},
|
||||
api_key: {
|
||||
id: 'api_key',
|
||||
label: this.$t('assets.ApiKey'),
|
||||
component: UploadSecret,
|
||||
hidden: (formValue) => formValue.secret_type !== 'api_key' || this.addTemplate
|
||||
},
|
||||
secret_type: {
|
||||
type: 'radio-group',
|
||||
options: [],
|
||||
hidden: () => {
|
||||
return this.addTemplate
|
||||
}
|
||||
},
|
||||
push_now: {
|
||||
helpText: this.$t('accounts.AccountPush.WindowsPushHelpText'),
|
||||
hidden: () => {
|
||||
const automation = this.iPlatform.automation || {}
|
||||
return !automation.push_account_enabled ||
|
||||
!automation.ansible_enabled ||
|
||||
!this.$hasPerm('accounts.push_account') ||
|
||||
this.addTemplate
|
||||
}
|
||||
},
|
||||
params: {
|
||||
label: this.$t('assets.PushParams'),
|
||||
component: AutomationParamsForm,
|
||||
el: {
|
||||
method: this.asset?.auto_config?.push_account_method
|
||||
},
|
||||
hidden: (formValue) => {
|
||||
const automation = this.iPlatform.automation || {}
|
||||
return !formValue.push_now ||
|
||||
!automation.push_account_enabled ||
|
||||
!automation.ansible_enabled ||
|
||||
!this.$hasPerm('accounts.push_account') ||
|
||||
this.addTemplate
|
||||
}
|
||||
},
|
||||
comment: {
|
||||
hidden: () => {
|
||||
return this.addTemplate
|
||||
}
|
||||
}
|
||||
},
|
||||
hasSaveContinue: false
|
||||
}
|
||||
},
|
||||
async mounted() {
|
||||
try {
|
||||
await this.getPlatform()
|
||||
this.setSecretTypeOptions()
|
||||
} finally {
|
||||
this.loading = false
|
||||
}
|
||||
},
|
||||
methods: {
|
||||
async getPlatform() {
|
||||
if (this.platform) {
|
||||
this.iPlatform = this.platform
|
||||
}
|
||||
if (!this.asset || !this.asset.platform) {
|
||||
return
|
||||
}
|
||||
const platformId = this.asset.platform.id
|
||||
this.iPlatform = await this.$axios.get(`/api/v1/assets/platforms/${platformId}/`)
|
||||
},
|
||||
setSecretTypeOptions() {
|
||||
const choices = [
|
||||
{
|
||||
label: this.$t('assets.Password'),
|
||||
value: 'password'
|
||||
},
|
||||
{
|
||||
label: this.$t('assets.SSHKey'),
|
||||
value: 'ssh_key'
|
||||
},
|
||||
{
|
||||
label: this.$t('assets.Token'),
|
||||
value: 'token'
|
||||
},
|
||||
{
|
||||
label: this.$t('assets.AccessKey'),
|
||||
value: 'access_key'
|
||||
},
|
||||
{
|
||||
label: this.$t('assets.ApiKey'),
|
||||
value: 'api_key'
|
||||
}
|
||||
]
|
||||
const secretTypes = []
|
||||
this.iPlatform.protocols?.forEach(p => {
|
||||
secretTypes.push(...p['secret_types'])
|
||||
})
|
||||
if (!this.form?.secret_type) {
|
||||
this.form.secret_type = secretTypes[0]
|
||||
}
|
||||
this.fieldsMeta.secret_type.options = choices.filter(item => {
|
||||
return secretTypes.indexOf(item.value) > -1
|
||||
})
|
||||
},
|
||||
confirm(form) {
|
||||
const secretType = form.secret_type || 'password'
|
||||
form.secret = form[secretType]
|
||||
form.secret = this.encryptPassword ? encryptPassword(form.secret) : form.secret
|
||||
|
||||
if (!form.secret) {
|
||||
delete form['secret']
|
||||
}
|
||||
if (this.account?.name) {
|
||||
this.$emit('edit', form)
|
||||
} else {
|
||||
this.$emit('add', form)
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
</script>
|
||||
|
||||
<style scoped>
|
||||
</style>
|
||||
@@ -1,156 +0,0 @@
|
||||
<template>
|
||||
<div>
|
||||
<GenericListPage :table-config="tableConfig" :header-actions="headerActions" :help-message="title" />
|
||||
<ShowSecretInfo v-if="showViewSecretDialog" :visible.sync="showViewSecretDialog" :account="account" />
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<script>
|
||||
import { ActionsFormatter, DetailFormatter, DisplayFormatter } from '@/components/TableFormatters'
|
||||
import ShowSecretInfo from '../AccountListTable/ShowSecretInfo'
|
||||
import { GenericListPage } from '@/layout/components'
|
||||
|
||||
export default {
|
||||
name: 'AccountHistoryTable',
|
||||
components: {
|
||||
GenericListPage,
|
||||
ShowSecretInfo
|
||||
},
|
||||
props: {
|
||||
url: {
|
||||
type: String,
|
||||
required: true
|
||||
},
|
||||
exportUrl: {
|
||||
type: String,
|
||||
default() {
|
||||
return this.url.replace('/assets/accounts-history/', '/assets/account-history-secrets/')
|
||||
}
|
||||
},
|
||||
hasLeftActions: {
|
||||
type: Boolean,
|
||||
default: false
|
||||
},
|
||||
otherActions: {
|
||||
type: Array,
|
||||
default: null
|
||||
},
|
||||
hasClone: {
|
||||
type: Boolean,
|
||||
default: false
|
||||
}
|
||||
},
|
||||
data() {
|
||||
const vm = this
|
||||
return {
|
||||
showViewSecretDialog: false,
|
||||
account: {},
|
||||
tableConfig: {
|
||||
url: this.url,
|
||||
permissions: {
|
||||
app: 'assets',
|
||||
resource: 'authbook'
|
||||
},
|
||||
columns: [
|
||||
'hostname', 'ip', 'username', 'version',
|
||||
'systemuser', 'date_created', 'date_updated', 'actions'
|
||||
],
|
||||
columnsShow: {
|
||||
min: ['username', 'actions'],
|
||||
default: ['hostname', 'ip', 'username', 'version', 'actions']
|
||||
},
|
||||
columnsMeta: {
|
||||
hostname: {
|
||||
prop: 'hostname',
|
||||
label: this.$t('assets.Hostname'),
|
||||
showOverflowTooltip: true,
|
||||
formatter: DetailFormatter,
|
||||
formatterArgs: {
|
||||
can: this.$hasPerm('assets.view_asset'),
|
||||
getRoute({ row }) {
|
||||
return {
|
||||
name: 'AssetDetail',
|
||||
params: { id: row.asset }
|
||||
}
|
||||
}
|
||||
}
|
||||
},
|
||||
ip: {
|
||||
width: '120px'
|
||||
},
|
||||
username: {
|
||||
showOverflowTooltip: true
|
||||
},
|
||||
systemuser: {
|
||||
formatter: DisplayFormatter
|
||||
},
|
||||
version: {
|
||||
width: '70px'
|
||||
},
|
||||
actions: {
|
||||
formatter: ActionsFormatter,
|
||||
formatterArgs: {
|
||||
hasUpdate: false, // can set function(row, value)
|
||||
hasDelete: false, // can set function(row, value)
|
||||
hasClone: this.hasClone,
|
||||
moreActionsTitle: this.$t('common.More'),
|
||||
extraActions: [
|
||||
{
|
||||
name: 'View',
|
||||
title: this.$t('common.View'),
|
||||
can: this.$hasPerm('assets.view_assethistoryaccountsecret'),
|
||||
type: 'primary',
|
||||
callback: ({ row }) => {
|
||||
vm.account = row
|
||||
vm.showViewSecretDialog = true
|
||||
}
|
||||
}
|
||||
]
|
||||
}
|
||||
}
|
||||
}
|
||||
},
|
||||
headerActions: {
|
||||
hasLeftActions: this.hasLeftActions,
|
||||
hasMoreActions: false,
|
||||
hasCreate: false,
|
||||
hasImport: false,
|
||||
hasExport: this.$hasPerm('assets.view_assethistoryaccountsecret'),
|
||||
exportOptions: {
|
||||
url: this.exportUrl,
|
||||
mfaVerifyRequired: true
|
||||
},
|
||||
searchConfig: {
|
||||
exclude: ['systemuser', 'asset']
|
||||
},
|
||||
hasSearch: true
|
||||
}
|
||||
}
|
||||
},
|
||||
computed: {
|
||||
title() {
|
||||
return this.$t('accounts.AccountHistableHelpMessage')
|
||||
}
|
||||
},
|
||||
watch: {
|
||||
url(iNew) {
|
||||
this.$set(this.tableConfig, 'url', iNew)
|
||||
this.$set(this.headerActions.exportOptions, 'url', iNew.replace('/accounts-history/', '/account-history-secrets/'))
|
||||
}
|
||||
},
|
||||
mounted() {
|
||||
if (this.otherActions) {
|
||||
const actionColumn = this.tableConfig.columns[this.tableConfig.columns.length - 1]
|
||||
for (const item of this.otherActions) {
|
||||
actionColumn.formatterArgs.extraActions.push(item)
|
||||
}
|
||||
}
|
||||
},
|
||||
methods: {
|
||||
}
|
||||
}
|
||||
</script>
|
||||
|
||||
<style lang='less' scoped>
|
||||
|
||||
</style>
|
||||
178
src/components/AccountListTable/AccountCreateUpdate.vue
Normal file
178
src/components/AccountListTable/AccountCreateUpdate.vue
Normal file
@@ -0,0 +1,178 @@
|
||||
<template>
|
||||
<Dialog
|
||||
v-if="iVisible"
|
||||
:close-on-click-modal="false"
|
||||
:destroy-on-close="true"
|
||||
:show-cancel="false"
|
||||
:show-confirm="false"
|
||||
:title="title"
|
||||
:visible.sync="iVisible"
|
||||
v-bind="$attrs"
|
||||
width="70%"
|
||||
v-on="$listeners"
|
||||
>
|
||||
<AccountCreateUpdateForm
|
||||
v-if="!loading"
|
||||
ref="form"
|
||||
:account="account"
|
||||
:add-template="addTemplate"
|
||||
:asset="asset"
|
||||
@add="addAccount"
|
||||
@edit="editAccount"
|
||||
/>
|
||||
</Dialog>
|
||||
</template>
|
||||
|
||||
<script>
|
||||
import Dialog from '@/components/Dialog'
|
||||
import AccountCreateUpdateForm from '@/components/AccountCreateUpdateForm'
|
||||
|
||||
export default {
|
||||
name: 'CreateAccountDialog',
|
||||
components: {
|
||||
Dialog,
|
||||
AccountCreateUpdateForm
|
||||
},
|
||||
props: {
|
||||
visible: {
|
||||
type: Boolean,
|
||||
default: false
|
||||
},
|
||||
addTemplate: {
|
||||
type: Boolean,
|
||||
default: false
|
||||
},
|
||||
asset: {
|
||||
type: Object,
|
||||
default: null
|
||||
},
|
||||
account: {
|
||||
type: Object,
|
||||
default: () => ({})
|
||||
},
|
||||
title: {
|
||||
type: String,
|
||||
default: function() {
|
||||
return this.$t('assets.AddAccount')
|
||||
}
|
||||
}
|
||||
},
|
||||
data() {
|
||||
return {
|
||||
loading: false,
|
||||
platform: {}
|
||||
}
|
||||
},
|
||||
computed: {
|
||||
protocols() {
|
||||
return this.asset ? this.asset.protocol : []
|
||||
},
|
||||
iVisible: {
|
||||
get() {
|
||||
return this.visible
|
||||
},
|
||||
set(val) {
|
||||
this.$emit('update:visible', val)
|
||||
}
|
||||
}
|
||||
},
|
||||
methods: {
|
||||
addAccount(form) {
|
||||
const formValue = Object.assign({}, form)
|
||||
let data, url, iVisible
|
||||
if (this.asset) {
|
||||
data = {
|
||||
asset: this.asset.id,
|
||||
...formValue
|
||||
}
|
||||
iVisible = false
|
||||
url = `/api/v1/accounts/accounts/`
|
||||
} else {
|
||||
iVisible = true
|
||||
data = formValue
|
||||
url = `/api/v1/accounts/accounts/bulk/`
|
||||
if (data.assets.length === 0) {
|
||||
this.$message.error(this.$tc('assets.PleaseSelectAsset'))
|
||||
return
|
||||
}
|
||||
}
|
||||
this.$axios.post(url, data, {
|
||||
disableFlashErrorMsg: iVisible
|
||||
}).then((data) => {
|
||||
this.handleResult(data, null)
|
||||
this.iVisible = iVisible
|
||||
if (!iVisible) {
|
||||
this.$emit('add', true)
|
||||
}
|
||||
}).catch(error => {
|
||||
this.iVisible = true
|
||||
this.handleResult(null, error)
|
||||
})
|
||||
},
|
||||
editAccount(form) {
|
||||
const data = { ...form }
|
||||
this.$axios.patch(`/api/v1/accounts/accounts/${this.account.id}/`, data).then(() => {
|
||||
this.iVisible = false
|
||||
this.$emit('add', true)
|
||||
this.$message.success(this.$tc('common.updateSuccessMsg'))
|
||||
}).catch(error => this.setFieldError(error))
|
||||
},
|
||||
handleResult(resp, error) {
|
||||
let bulkCreate = !this.asset
|
||||
if (error && !Array.isArray(error?.response?.data)) {
|
||||
bulkCreate = false
|
||||
}
|
||||
if (resp && !Array.isArray(resp)) {
|
||||
bulkCreate = false
|
||||
}
|
||||
if (!bulkCreate) {
|
||||
if (!error) {
|
||||
this.$message.success(this.$tc('common.createSuccessMsg'))
|
||||
} else {
|
||||
this.setFieldError(error)
|
||||
}
|
||||
} else {
|
||||
let result
|
||||
if (error) {
|
||||
result = error.response.data
|
||||
} else {
|
||||
result = resp
|
||||
}
|
||||
this.$emit('bulk-create-done', result)
|
||||
}
|
||||
},
|
||||
setFieldError(error) {
|
||||
const response = error.response
|
||||
const data = response.data
|
||||
const refsAutoDataForm = this.$refs.form.$refs.AutoDataForm
|
||||
if (response.status === 400) {
|
||||
for (const key of Object.keys(data)) {
|
||||
let err = ''
|
||||
let current = key
|
||||
let errorTips = data[current]
|
||||
if (errorTips instanceof Array) {
|
||||
errorTips = _.filter(errorTips, (item) => Object.keys(item).length > 0)
|
||||
for (const i of errorTips) {
|
||||
if (i instanceof Object) {
|
||||
err += i?.port?.join(',')
|
||||
} else {
|
||||
err += errorTips
|
||||
}
|
||||
}
|
||||
} else {
|
||||
err = errorTips
|
||||
}
|
||||
if (current === 'secret') {
|
||||
current = refsAutoDataForm.form.secret_type?.value || key
|
||||
}
|
||||
refsAutoDataForm.setFieldError(current, err)
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
</script>
|
||||
|
||||
<style scoped>
|
||||
|
||||
</style>
|
||||
438
src/components/AccountListTable/AccountList.vue
Normal file
438
src/components/AccountListTable/AccountList.vue
Normal file
@@ -0,0 +1,438 @@
|
||||
<template>
|
||||
<div>
|
||||
<ListTable ref="ListTable" :header-actions="headerActions" :table-config="tableConfig" />
|
||||
<ViewSecret
|
||||
v-if="showViewSecretDialog"
|
||||
:account="account"
|
||||
:url="secretUrl"
|
||||
:visible.sync="showViewSecretDialog"
|
||||
/>
|
||||
<UpdateSecretInfo
|
||||
v-if="showUpdateSecretDialog"
|
||||
:account="account"
|
||||
:visible.sync="showUpdateSecretDialog"
|
||||
@updateAuthDone="onUpdateAuthDone"
|
||||
/>
|
||||
<AccountCreateUpdate
|
||||
v-if="showAddDialog"
|
||||
:account="account"
|
||||
:asset="iAsset"
|
||||
:title="accountCreateUpdateTitle"
|
||||
:visible.sync="showAddDialog"
|
||||
@add="addAccountSuccess"
|
||||
@bulk-create-done="showBulkCreateResult($event)"
|
||||
/>
|
||||
<AccountCreateUpdate
|
||||
v-if="showAddTemplateDialog"
|
||||
:account="account"
|
||||
:add-template="true"
|
||||
:asset="iAsset"
|
||||
:title="accountCreateUpdateTitle"
|
||||
:visible.sync="showAddTemplateDialog"
|
||||
@add="addAccountSuccess"
|
||||
@bulk-create-done="showBulkCreateResult($event)"
|
||||
/>
|
||||
<ResultDialog
|
||||
v-if="showResultDialog"
|
||||
:result="createAccountResults"
|
||||
:visible.sync="showResultDialog"
|
||||
/>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<script>
|
||||
import ListTable from '@/components/ListTable/index'
|
||||
import { ActionsFormatter } from '@/components/TableFormatters'
|
||||
import ViewSecret from './ViewSecret'
|
||||
import UpdateSecretInfo from './UpdateSecretInfo'
|
||||
import AccountCreateUpdate from './AccountCreateUpdate'
|
||||
import { connectivityMeta } from './const'
|
||||
import { openTaskPage } from '@/utils/jms'
|
||||
import ResultDialog from './BulkCreateResultDialog.vue'
|
||||
|
||||
export default {
|
||||
name: 'AccountListTable',
|
||||
components: {
|
||||
ResultDialog,
|
||||
ListTable,
|
||||
UpdateSecretInfo,
|
||||
ViewSecret,
|
||||
AccountCreateUpdate
|
||||
},
|
||||
props: {
|
||||
url: {
|
||||
type: String,
|
||||
required: true
|
||||
},
|
||||
exportUrl: {
|
||||
type: String,
|
||||
default() {
|
||||
return this.url.replace('/accounts/accounts/', '/accounts/account-secrets/')
|
||||
}
|
||||
},
|
||||
hasLeftActions: {
|
||||
type: Boolean,
|
||||
default: false
|
||||
},
|
||||
otherActions: {
|
||||
type: Array,
|
||||
default: null
|
||||
},
|
||||
hasClone: {
|
||||
type: Boolean,
|
||||
default: false
|
||||
},
|
||||
asset: {
|
||||
type: Object,
|
||||
default: null
|
||||
},
|
||||
columns: {
|
||||
type: Array,
|
||||
default: () => []
|
||||
},
|
||||
hasExport: {
|
||||
type: Boolean,
|
||||
default: true
|
||||
},
|
||||
hasImport: {
|
||||
type: Boolean,
|
||||
default: true
|
||||
},
|
||||
hasDeleteAction: {
|
||||
type: Boolean,
|
||||
default: true
|
||||
},
|
||||
columnsMeta: {
|
||||
type: Object,
|
||||
default: () => {
|
||||
}
|
||||
},
|
||||
columnsDefault: {
|
||||
type: Array,
|
||||
default: () => ([
|
||||
'name', 'username', 'asset', 'privileged',
|
||||
'secret_type', 'date_updated'
|
||||
])
|
||||
},
|
||||
headerExtraActions: {
|
||||
type: Array,
|
||||
default: () => []
|
||||
}
|
||||
},
|
||||
data() {
|
||||
const vm = this
|
||||
return {
|
||||
showViewSecretDialog: false,
|
||||
showUpdateSecretDialog: false,
|
||||
showResultDialog: false,
|
||||
showAddDialog: false,
|
||||
showAddTemplateDialog: false,
|
||||
createAccountResults: [],
|
||||
accountCreateUpdateTitle: this.$t('assets.AddAccount'),
|
||||
iAsset: this.asset,
|
||||
account: {},
|
||||
secretUrl: '',
|
||||
tableConfig: {
|
||||
url: this.url,
|
||||
permissions: {
|
||||
app: 'assets',
|
||||
resource: 'account'
|
||||
},
|
||||
extraQuery: {
|
||||
order: '-date_updated'
|
||||
},
|
||||
columnsExclude: ['spec_info'],
|
||||
columns: [
|
||||
'name', 'username', 'asset', 'privileged',
|
||||
'secret_type', 'source', 'actions'
|
||||
],
|
||||
columnsShow: {
|
||||
min: ['name', 'username', 'actions'],
|
||||
default: this.columnsDefault
|
||||
},
|
||||
columnsMeta: {
|
||||
name: {
|
||||
formatter: function(row) {
|
||||
const to = {
|
||||
name: 'AssetAccountDetail',
|
||||
params: { id: row.id }
|
||||
}
|
||||
if (vm.$hasPerm('accounts.view_account')) {
|
||||
return <router-link to={to}>{row.name}</router-link>
|
||||
} else {
|
||||
return <span>{row.name}</span>
|
||||
}
|
||||
}
|
||||
},
|
||||
asset: {
|
||||
label: this.$t('assets.Asset'),
|
||||
formatter: function(row) {
|
||||
const to = {
|
||||
name: 'AssetDetail',
|
||||
params: { id: row.asset.id }
|
||||
}
|
||||
if (vm.$hasPerm('assets.view_asset')) {
|
||||
return <router-link to={to}>{row.asset.name}</router-link>
|
||||
} else {
|
||||
return <span>{row.asset.name}</span>
|
||||
}
|
||||
}
|
||||
},
|
||||
secret_type: {
|
||||
width: '100px',
|
||||
formatter: function(row) {
|
||||
return row.secret_type.label
|
||||
}
|
||||
},
|
||||
source: {
|
||||
formatter: function(row) {
|
||||
return row.source.label
|
||||
}
|
||||
},
|
||||
has_secret: {
|
||||
width: '100px',
|
||||
formatterArgs: {
|
||||
showFalse: false
|
||||
}
|
||||
},
|
||||
privileged: {
|
||||
label: this.$t('assets.Privileged'),
|
||||
width: '120px',
|
||||
formatterArgs: {
|
||||
showText: false,
|
||||
showFalse: false
|
||||
}
|
||||
},
|
||||
connectivity: connectivityMeta,
|
||||
actions: {
|
||||
formatter: ActionsFormatter,
|
||||
formatterArgs: {
|
||||
hasUpdate: false, // can set function(row, value)
|
||||
hasDelete: false, // can set function(row, value)
|
||||
hasClone: this.hasClone,
|
||||
moreActionsTitle: this.$t('common.More'),
|
||||
extraActions: [
|
||||
{
|
||||
name: 'View',
|
||||
title: this.$t('common.View'),
|
||||
can: this.$hasPerm('accounts.view_accountsecret'),
|
||||
type: 'primary',
|
||||
callback: ({ row }) => {
|
||||
// debugger
|
||||
vm.secretUrl = `/api/v1/accounts/account-secrets/${row.id}/`
|
||||
vm.account = row
|
||||
vm.showViewSecretDialog = false
|
||||
setTimeout(() => {
|
||||
vm.showViewSecretDialog = true
|
||||
})
|
||||
}
|
||||
},
|
||||
{
|
||||
name: 'ClearSecret',
|
||||
title: this.$t('common.ClearSecret'),
|
||||
can: this.$hasPerm('accounts.change_account'),
|
||||
type: 'primary',
|
||||
callback: ({ row }) => {
|
||||
this.$axios.patch(
|
||||
`/api/v1/accounts/accounts/clear-secret/`,
|
||||
{ account_ids: [row.id] }
|
||||
).then(() => {
|
||||
this.$message.success(this.$tc('common.ClearSuccessMsg'))
|
||||
})
|
||||
}
|
||||
},
|
||||
{
|
||||
name: 'Test',
|
||||
title: this.$t('common.Test'),
|
||||
can: ({ row }) =>
|
||||
!this.$store.getters.currentOrgIsRoot &&
|
||||
this.$hasPerm('accounts.change_account') &&
|
||||
row.asset['auto_config'].ansible_enabled &&
|
||||
row.asset['auto_config'].ping_enabled,
|
||||
callback: ({ row }) => {
|
||||
this.$axios.post(
|
||||
`/api/v1/accounts/accounts/tasks/`,
|
||||
{ action: 'verify', accounts: [row.id] }
|
||||
).then(res => {
|
||||
openTaskPage(res['task'])
|
||||
})
|
||||
}
|
||||
},
|
||||
{
|
||||
name: 'Update',
|
||||
title: this.$t('common.Update'),
|
||||
can: this.$hasPerm('accounts.change_account') && !this.$store.getters.currentOrgIsRoot,
|
||||
callback: ({ row }) => {
|
||||
const data = {
|
||||
...this.asset,
|
||||
...row.asset
|
||||
}
|
||||
vm.account = row
|
||||
vm.iAsset = data
|
||||
vm.showAddDialog = false
|
||||
vm.accountCreateUpdateTitle = this.$t('assets.UpdateAccount')
|
||||
setTimeout(() => {
|
||||
vm.showAddDialog = true
|
||||
})
|
||||
}
|
||||
}
|
||||
]
|
||||
}
|
||||
},
|
||||
...this.columnsMeta
|
||||
}
|
||||
},
|
||||
headerActions: {
|
||||
hasLeftActions: this.hasLeftActions,
|
||||
hasMoreActions: true,
|
||||
hasCreate: false,
|
||||
hasImport: this.hasImport,
|
||||
hasExport: this.hasExport && this.$hasPerm('accounts.view_accountsecret'),
|
||||
handleImportClick: ({ selectedRows }) => {
|
||||
this.$eventBus.$emit('showImportDialog', {
|
||||
selectedRows,
|
||||
url: '/api/v1/accounts/accounts/',
|
||||
name: this?.name
|
||||
})
|
||||
},
|
||||
exportOptions: {
|
||||
url: this.exportUrl,
|
||||
mfaVerifyRequired: true,
|
||||
tips: this.$t('accounts.AccountExportTips')
|
||||
},
|
||||
importOptions: {
|
||||
canImportCreate: this.$hasPerm('accounts.add_account'),
|
||||
canImportUpdate: this.$hasPerm('accounts.change_account')
|
||||
},
|
||||
extraActions: [
|
||||
{
|
||||
name: 'add',
|
||||
title: this.$t('common.Add'),
|
||||
type: 'primary',
|
||||
can: () => {
|
||||
return vm.$hasPerm('accounts.add_account') && !this.$store.getters.currentOrgIsRoot
|
||||
},
|
||||
callback: async() => {
|
||||
await this.getAssetDetail()
|
||||
setTimeout(() => {
|
||||
vm.iAsset = this.asset
|
||||
vm.account = {}
|
||||
vm.accountCreateUpdateTitle = this.$t('assets.AddAccount')
|
||||
vm.showAddDialog = true
|
||||
})
|
||||
}
|
||||
},
|
||||
{
|
||||
name: 'add-template',
|
||||
title: this.$t('common.TemplateAdd'),
|
||||
has: !(this.platform || this.asset),
|
||||
can: () => {
|
||||
return vm.$hasPerm('accounts.add_account') && !this.$store.getters.currentOrgIsRoot
|
||||
},
|
||||
callback: async() => {
|
||||
await this.getAssetDetail()
|
||||
setTimeout(() => {
|
||||
vm.iAsset = this.asset
|
||||
vm.account = {}
|
||||
vm.accountCreateUpdateTitle = this.$t('assets.AddAccount')
|
||||
vm.showAddTemplateDialog = true
|
||||
})
|
||||
}
|
||||
},
|
||||
...this.headerExtraActions
|
||||
],
|
||||
extraMoreActions: [
|
||||
{
|
||||
name: 'ClearSecrets',
|
||||
title: this.$t('common.ClearSecret'),
|
||||
type: 'primary',
|
||||
fa: 'clean',
|
||||
can: ({ selectedRows }) => {
|
||||
return selectedRows.length > 0 && vm.$hasPerm('accounts.change_account')
|
||||
},
|
||||
callback: function({ selectedRows }) {
|
||||
const ids = selectedRows.map(v => { return v.id })
|
||||
this.$axios.patch(
|
||||
'/api/v1/accounts/accounts/clear-secret/',
|
||||
{ account_ids: ids }).then(() => {
|
||||
this.$message.success(this.$tc('common.ClearSuccessMsg'))
|
||||
}).catch(err => {
|
||||
this.$message.error(this.$tc('common.bulkClearErrorMsg' + ' ' + err))
|
||||
})
|
||||
}.bind(this)
|
||||
}
|
||||
],
|
||||
canBulkDelete: vm.$hasPerm('accounts.delete_account'),
|
||||
searchConfig: {
|
||||
getUrlQuery: false,
|
||||
exclude: ['asset']
|
||||
},
|
||||
hasSearch: true
|
||||
}
|
||||
}
|
||||
},
|
||||
watch: {
|
||||
url(iNew) {
|
||||
this.$set(this.tableConfig, 'url', iNew)
|
||||
this.$set(this.headerActions.exportOptions, 'url', iNew.replace(/(.*)accounts/, '$1account-secrets'))
|
||||
}
|
||||
},
|
||||
mounted() {
|
||||
if (this.columns.length > 0) {
|
||||
this.tableConfig.columns = this.columns
|
||||
}
|
||||
if (this.otherActions) {
|
||||
const actionColumn = this.tableConfig.columns[this.tableConfig.columns.length - 1]
|
||||
for (const item of this.otherActions) {
|
||||
actionColumn.formatterArgs.extraActions.push(item)
|
||||
}
|
||||
}
|
||||
if (this.hasDeleteAction) {
|
||||
this.tableConfig.columnsMeta.actions.formatterArgs.extraActions.push(
|
||||
{
|
||||
name: 'Delete',
|
||||
title: this.$t('common.Delete'),
|
||||
can: this.$hasPerm('accounts.delete_account'),
|
||||
type: 'primary',
|
||||
callback: ({ row }) => {
|
||||
this.$axios.delete(`/api/v1/accounts/accounts/${row.id}/`).then(() => {
|
||||
this.$message.success(this.$tc('common.deleteSuccessMsg'))
|
||||
this.$refs.ListTable.reloadTable()
|
||||
})
|
||||
}
|
||||
}
|
||||
)
|
||||
}
|
||||
},
|
||||
methods: {
|
||||
onUpdateAuthDone(account) {
|
||||
Object.assign(this.account, account)
|
||||
},
|
||||
addAccountSuccess() {
|
||||
this.$refs.ListTable.reloadTable()
|
||||
},
|
||||
async getAssetDetail() {
|
||||
const { query: { asset }} = this.$route
|
||||
if (asset) {
|
||||
this.iAsset = await this.$axios.get(`/api/v1/assets/assets/${asset}/`)
|
||||
}
|
||||
},
|
||||
refresh() {
|
||||
this.$refs.ListTable.reloadTable()
|
||||
},
|
||||
showBulkCreateResult(results) {
|
||||
this.showResultDialog = false
|
||||
this.createAccountResults = results
|
||||
setTimeout(() => {
|
||||
this.showResultDialog = true
|
||||
}, 100)
|
||||
}
|
||||
}
|
||||
}
|
||||
</script>
|
||||
|
||||
<style lang='scss' scoped>
|
||||
.cell a {
|
||||
color: var(--color-info);
|
||||
}
|
||||
</style>
|
||||
120
src/components/AccountListTable/BulkCreateResultDialog.vue
Normal file
120
src/components/AccountListTable/BulkCreateResultDialog.vue
Normal file
@@ -0,0 +1,120 @@
|
||||
<template>
|
||||
<Dialog
|
||||
:show-cancel="false"
|
||||
:title="title"
|
||||
v-bind="$attrs"
|
||||
@confirm="closeDialog"
|
||||
v-on="$listeners"
|
||||
>
|
||||
<el-alert style="margin-bottom: 10px" type="success">
|
||||
<span v-for="item of summary" :key="item.key"><b>{{ item.label }}</b>: {{ item.value }} </span>
|
||||
</el-alert>
|
||||
<DataTable :config="config" />
|
||||
</Dialog>
|
||||
</template>
|
||||
|
||||
<script>
|
||||
import Dialog from '@/components/Dialog/index.vue'
|
||||
import DataTable from '@/components/DataTable/index.vue'
|
||||
|
||||
export default {
|
||||
name: 'ResultDialog',
|
||||
components: {
|
||||
DataTable,
|
||||
Dialog
|
||||
},
|
||||
props: {
|
||||
result: {
|
||||
type: Array,
|
||||
default: () => []
|
||||
}
|
||||
},
|
||||
data() {
|
||||
const errorProp = this.$t('common.Error')
|
||||
const stateMap = {
|
||||
'created': this.$tc('common.Created'),
|
||||
'updated': this.$tc('common.Updated'),
|
||||
'skipped': this.$tc('common.Skipped')
|
||||
}
|
||||
const stateClsMap = {
|
||||
'created': 'color-primary',
|
||||
'updated': 'color-success',
|
||||
'skipped': 'color-default'
|
||||
}
|
||||
return {
|
||||
title: this.$t('accounts.AddAccountResult'),
|
||||
config: {
|
||||
columns: [
|
||||
{
|
||||
prop: 'asset',
|
||||
label: this.$t('assets.Asset')
|
||||
},
|
||||
{
|
||||
prop: 'state',
|
||||
label: this.$t('common.Status'),
|
||||
width: '200px',
|
||||
formatter: (row) => {
|
||||
if (row.error) {
|
||||
return <span class='color-error'>{ errorProp }: { row.error }</span>
|
||||
} else if (row.state) {
|
||||
const colorCls = stateClsMap[row.state]
|
||||
const state = stateMap[row.state]
|
||||
return <span class={ colorCls }>{ state }</span>
|
||||
}
|
||||
}
|
||||
}
|
||||
],
|
||||
totalData: this.result
|
||||
}
|
||||
}
|
||||
},
|
||||
computed: {
|
||||
summary() {
|
||||
const labels = {
|
||||
total: this.$tc('common.Total'),
|
||||
created: this.$tc('common.Created'),
|
||||
updated: this.$tc('common.Updated'),
|
||||
skipped: this.$tc('common.Skipped'),
|
||||
error: this.$tc('common.Error')
|
||||
}
|
||||
const grouped = _.groupBy(this.result, 'state')
|
||||
const groupedLength = _.mapValues(grouped, 'length')
|
||||
groupedLength['total'] = this.result.length
|
||||
return _.map(groupedLength, (value, key) => {
|
||||
return {
|
||||
label: labels[key],
|
||||
value: value,
|
||||
key: key
|
||||
}
|
||||
})
|
||||
}
|
||||
},
|
||||
methods: {
|
||||
closeDialog() {
|
||||
this.$emit('update:visible', false)
|
||||
}
|
||||
}
|
||||
}
|
||||
</script>
|
||||
|
||||
<style scoped>
|
||||
.color-error {
|
||||
color: var(--color-danger);
|
||||
}
|
||||
|
||||
.color-primary {
|
||||
color: var(--color-primary);
|
||||
}
|
||||
|
||||
.color-success {
|
||||
color: var(--color-success);
|
||||
}
|
||||
|
||||
.color-default {
|
||||
}
|
||||
|
||||
::v-deep .el-data-table .el-table .el-table__row > td > div > span {
|
||||
white-space: inherit;
|
||||
}
|
||||
|
||||
</style>
|
||||
78
src/components/AccountListTable/PasswordHistoryDialog.vue
Normal file
78
src/components/AccountListTable/PasswordHistoryDialog.vue
Normal file
@@ -0,0 +1,78 @@
|
||||
<template>
|
||||
<GenericListTableDialog :visible.sync="iVisible" v-bind="config" />
|
||||
</template>
|
||||
|
||||
<script>
|
||||
import { GenericListTableDialog } from '@/layout/components'
|
||||
import { ShowKeyCopyFormatter } from '@/components/TableFormatters'
|
||||
|
||||
export default {
|
||||
components: {
|
||||
GenericListTableDialog
|
||||
},
|
||||
props: {
|
||||
account: {
|
||||
type: Object,
|
||||
default: () => ({})
|
||||
},
|
||||
visible: {
|
||||
type: Boolean,
|
||||
default: false
|
||||
}
|
||||
},
|
||||
data() {
|
||||
return {
|
||||
config: {
|
||||
title: this.$t('accounts.HistoryPassword'),
|
||||
visible: false,
|
||||
width: '60%',
|
||||
tableConfig: {
|
||||
id: 'history_date',
|
||||
url: `/api/v1/accounts/account-secrets/${this.account.id}/histories/`,
|
||||
columns: ['secret', 'secret_type', 'version', 'history_date'],
|
||||
columnsMeta: {
|
||||
secret: {
|
||||
label: this.$t('assets.Password'),
|
||||
formatter: ShowKeyCopyFormatter,
|
||||
formatterArgs: {
|
||||
hasDownload: false,
|
||||
name: this.account.name
|
||||
}
|
||||
},
|
||||
history_date: {
|
||||
label: this.$t('accounts.HistoryDate')
|
||||
},
|
||||
secret_type: {
|
||||
width: '200px'
|
||||
},
|
||||
version: {
|
||||
width: '100px'
|
||||
},
|
||||
actions: {
|
||||
has: false
|
||||
}
|
||||
}
|
||||
},
|
||||
headerActions: {
|
||||
hasLeftActions: false,
|
||||
hasSearch: false
|
||||
}
|
||||
}
|
||||
}
|
||||
},
|
||||
computed: {
|
||||
iVisible: {
|
||||
get() {
|
||||
return this.visible
|
||||
},
|
||||
set(val) {
|
||||
this.$emit('update:visible', val)
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
</script>
|
||||
|
||||
<style scoped>
|
||||
|
||||
</style>
|
||||
@@ -1,83 +0,0 @@
|
||||
<template>
|
||||
<div>
|
||||
<UserConfirmDialog
|
||||
:url="url"
|
||||
@UserConfirmDone="getAuthInfo"
|
||||
@UserConfirmCancel="exit"
|
||||
/>
|
||||
<Dialog
|
||||
:title="dialogTitle"
|
||||
:show-confirm="false"
|
||||
:show-cancel="false"
|
||||
:destroy-on-close="true"
|
||||
:width="'50'"
|
||||
:visible.sync="showAuthInfo"
|
||||
v-bind="$attrs"
|
||||
v-on="$listeners"
|
||||
>
|
||||
<div>
|
||||
<el-form label-position="right" label-width="80px" :model="authInfo">
|
||||
<el-form-item :label="this.$t('assets.Hostname')">
|
||||
<el-input v-model="account.hostname" readonly />
|
||||
</el-form-item>
|
||||
<el-form-item :label="this.$t('assets.Username')">
|
||||
<el-input v-model="account['username']" readonly />
|
||||
</el-form-item>
|
||||
<el-form-item :label="this.$t('assets.Password')">
|
||||
<el-input v-model="authInfo.password" type="password" show-password />
|
||||
</el-form-item>
|
||||
<el-form-item :label="this.$t('users.SSHKey')">
|
||||
<el-input v-model="authInfo['private_key']" class="item-textarea" type="textarea" show-password />
|
||||
</el-form-item>
|
||||
</el-form>
|
||||
</div>
|
||||
</Dialog>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<script>
|
||||
import Dialog from '@/components/Dialog'
|
||||
import UserConfirmDialog from '@/components/UserConfirmDialog'
|
||||
export default {
|
||||
name: 'ShowSecretInfo',
|
||||
components: {
|
||||
Dialog,
|
||||
UserConfirmDialog
|
||||
},
|
||||
props: {
|
||||
account: {
|
||||
type: Object,
|
||||
default: () => ({})
|
||||
},
|
||||
visible: {
|
||||
type: Boolean,
|
||||
default: false
|
||||
}
|
||||
},
|
||||
data() {
|
||||
return {
|
||||
dialogTitle: this.$t('common.ViewSecret'),
|
||||
authInfo: {},
|
||||
showAuthInfo: false,
|
||||
url: `/api/v1/assets/account-secrets/${this.account.id}/`
|
||||
}
|
||||
},
|
||||
methods: {
|
||||
getAuthInfo() {
|
||||
this.$axios.get(this.url, { disableFlashErrorMsg: true }).then(resp => {
|
||||
this.authInfo = resp
|
||||
this.showAuthInfo = true
|
||||
})
|
||||
},
|
||||
exit() {
|
||||
this.$emit('update:visible', false)
|
||||
}
|
||||
}
|
||||
}
|
||||
</script>
|
||||
|
||||
<style scoped>
|
||||
.item-textarea >>> .el-textarea__inner {
|
||||
height: 110px;
|
||||
}
|
||||
</style>
|
||||
@@ -1,27 +1,27 @@
|
||||
<template>
|
||||
<Dialog
|
||||
width="50"
|
||||
:title="this.$t('assets.UpdateAssetUserToken')"
|
||||
:visible.sync="visible"
|
||||
:destroy-on-close="true"
|
||||
@confirm="handleConfirm()"
|
||||
:title="$tc('assets.UpdateAssetUserToken')"
|
||||
:visible.sync="visible"
|
||||
width="50"
|
||||
@cancel="handleCancel()"
|
||||
@confirm="handleConfirm()"
|
||||
v-on="$listeners"
|
||||
>
|
||||
<el-form label-position="right" label-width="90px">
|
||||
<el-form-item :label="this.$t('assets.Hostname')">
|
||||
<el-input v-model="account.hostname" readonly />
|
||||
<el-form-item :label="$tc('assets.Name')">
|
||||
<el-input v-model="account['asset_name']" readonly />
|
||||
</el-form-item>
|
||||
<el-form-item :label="this.$t('assets.Username')">
|
||||
<el-form-item :label="$tc('assets.Username')">
|
||||
<el-input v-model="account['username']" readonly />
|
||||
</el-form-item>
|
||||
<el-form-item :label="this.$t('assets.Password')">
|
||||
<el-form-item :label="$tc('assets.Password')">
|
||||
<UpdateToken v-model="authInfo.password" />
|
||||
</el-form-item>
|
||||
<el-form-item :label="this.$t('assets.SSHSecretKey')">
|
||||
<el-form-item :label="$tc('assets.SSHSecretKey')">
|
||||
<UploadKey @input="getFile" />
|
||||
</el-form-item>
|
||||
<el-form-item :label="this.$t('assets.Passphrase')">
|
||||
<el-form-item :label="$tc('assets.Passphrase')">
|
||||
<UpdateToken v-model="authInfo.passphrase" />
|
||||
</el-form-item>
|
||||
</el-form>
|
||||
@@ -32,6 +32,7 @@
|
||||
import Dialog from '@/components/Dialog'
|
||||
import { UpdateToken, UploadKey } from '@/components/FormFields'
|
||||
import { encryptPassword } from '@/utils/crypto'
|
||||
|
||||
export default {
|
||||
name: 'UpdateSecretInfo',
|
||||
components: {
|
||||
@@ -51,7 +52,7 @@ export default {
|
||||
},
|
||||
data() {
|
||||
return {
|
||||
authInfo: {
|
||||
secretInfo: {
|
||||
password: '',
|
||||
private_key: '',
|
||||
passphrase: ''
|
||||
@@ -61,15 +62,15 @@ export default {
|
||||
methods: {
|
||||
handleConfirm() {
|
||||
const data = {}
|
||||
if (this.authInfo.password !== '') {
|
||||
data.password = encryptPassword(this.authInfo.password)
|
||||
if (this.secretInfo.password !== '') {
|
||||
data.password = encryptPassword(this.secretInfo.password)
|
||||
}
|
||||
if (this.authInfo.private_key !== '') {
|
||||
data.private_key = encryptPassword(this.authInfo.private_key)
|
||||
if (this.authInfo.passphrase) data.passphrase = this.authInfo.passphrase
|
||||
if (this.secretInfo.private_key !== '') {
|
||||
data.private_key = encryptPassword(this.secretInfo.private_key)
|
||||
if (this.secretInfo.passphrase) data.passphrase = this.secretInfo.passphrase
|
||||
}
|
||||
this.$axios.patch(
|
||||
`/api/v1/assets/accounts/${this.account.id}/`,
|
||||
`/api/v1/accounts/accounts/${this.account.id}/`,
|
||||
data,
|
||||
{ disableFlashErrorMsg: true }
|
||||
).then(res => {
|
||||
@@ -87,7 +88,7 @@ export default {
|
||||
this.$emit('update:visible', false)
|
||||
},
|
||||
getFile(file) {
|
||||
this.authInfo.private_key = file
|
||||
this.secretInfo.private_key = file
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
215
src/components/AccountListTable/ViewSecret.vue
Normal file
215
src/components/AccountListTable/ViewSecret.vue
Normal file
@@ -0,0 +1,215 @@
|
||||
<template>
|
||||
<div>
|
||||
<div v-if="mfaDialogVisible">
|
||||
<UserConfirmDialog
|
||||
:url="url"
|
||||
@UserConfirmCancel="exit"
|
||||
@UserConfirmDone="getAuthInfo"
|
||||
/>
|
||||
</div>
|
||||
<Dialog
|
||||
:destroy-on-close="true"
|
||||
:show-cancel="false"
|
||||
:title="title"
|
||||
:visible.sync="showSecret"
|
||||
:width="'50'"
|
||||
v-bind="$attrs"
|
||||
@confirm="accountConfirmHandle"
|
||||
v-on="$listeners"
|
||||
>
|
||||
<el-form :model="secretInfo" class="password-form" label-position="right" label-width="100px">
|
||||
<el-form-item :label="$tc('assets.Name')">
|
||||
<span>{{ account['name'] }}</span>
|
||||
</el-form-item>
|
||||
<el-form-item :label="$tc('assets.Username')">
|
||||
<span>{{ account['username'] }}</span>
|
||||
</el-form-item>
|
||||
<el-form-item :label="secretTypeLabel">
|
||||
<ShowKeyCopyFormatter
|
||||
:cell-value="secretInfo.secret"
|
||||
:col="{ formatterArgs: {
|
||||
name: account['name'],
|
||||
secretType: secretType || ''
|
||||
}}"
|
||||
@input="onShowKeyCopyFormatterChange"
|
||||
/>
|
||||
</el-form-item>
|
||||
<el-form-item v-if="secretType === 'ssh_key'" :label="$tc('assets.sshKeyFingerprint')">
|
||||
<span>{{ sshKeyFingerprint }}</span>
|
||||
</el-form-item>
|
||||
<el-form-item :label="$tc('common.DateCreated')">
|
||||
<span>{{ account['date_created'] | date }}</span>
|
||||
</el-form-item>
|
||||
<el-form-item :label="$tc('common.DateUpdated')">
|
||||
<span>{{ account['date_updated'] | date }}</span>
|
||||
</el-form-item>
|
||||
<el-form-item v-if="showPasswordRecord" v-perms="'accounts.view_accountsecret'" :label="$tc('accounts.PasswordRecord')">
|
||||
<el-link
|
||||
:underline="false"
|
||||
type="success"
|
||||
@click="showHistoryDialog"
|
||||
>
|
||||
<span style="padding-right: 30px">
|
||||
{{ versions }}
|
||||
</span>
|
||||
</el-link>
|
||||
</el-form-item>
|
||||
</el-form>
|
||||
</Dialog>
|
||||
<PasswordHistoryDialog
|
||||
v-if="showPasswordHistoryDialog"
|
||||
:account="account"
|
||||
:visible.sync="showPasswordHistoryDialog"
|
||||
/>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<script>
|
||||
import Dialog from '@/components/Dialog'
|
||||
import PasswordHistoryDialog from './PasswordHistoryDialog'
|
||||
import UserConfirmDialog from '@/components/UserConfirmDialog'
|
||||
import { ShowKeyCopyFormatter } from '@/components/TableFormatters'
|
||||
import { encryptPassword } from '@/utils/crypto'
|
||||
|
||||
export default {
|
||||
name: 'ShowSecretInfo',
|
||||
components: {
|
||||
Dialog,
|
||||
PasswordHistoryDialog,
|
||||
UserConfirmDialog,
|
||||
ShowKeyCopyFormatter
|
||||
},
|
||||
props: {
|
||||
account: {
|
||||
type: Object,
|
||||
default: () => ({})
|
||||
},
|
||||
visible: {
|
||||
type: Boolean,
|
||||
default: false
|
||||
},
|
||||
url: {
|
||||
type: String,
|
||||
default: ''
|
||||
},
|
||||
title: {
|
||||
type: String,
|
||||
default: function() {
|
||||
return this.$tc('assets.AccountDetail')
|
||||
}
|
||||
},
|
||||
showPasswordRecord: {
|
||||
type: Boolean,
|
||||
default: true
|
||||
}
|
||||
},
|
||||
data() {
|
||||
return {
|
||||
modifiedSecret: '',
|
||||
secretInfo: {},
|
||||
versions: '-',
|
||||
showSecret: false,
|
||||
mfaDialogVisible: true,
|
||||
sshKeyFingerprint: '-',
|
||||
historyCount: 0,
|
||||
showPasswordHistoryDialog: false
|
||||
}
|
||||
},
|
||||
computed: {
|
||||
secretTypeLabel() {
|
||||
return this.account['secret_type'].label || 'Password'
|
||||
},
|
||||
secretType() {
|
||||
return this.account['secret_type'].value
|
||||
}
|
||||
},
|
||||
mounted() {
|
||||
if (this.showPasswordRecord) {
|
||||
const url = `/api/v1/accounts/account-secrets/${this.account.id}/histories/?limit=1`
|
||||
this.$axios.get(url, { disableFlashErrorMsg: true }).then(resp => {
|
||||
this.versions = resp.count
|
||||
})
|
||||
}
|
||||
},
|
||||
methods: {
|
||||
accountConfirmHandle() {
|
||||
this.modifiedSecret && this.onChangeSecretSubmit()
|
||||
this.showSecret = false
|
||||
this.mfaDialogVisible = false
|
||||
},
|
||||
onChangeSecretSubmit() {
|
||||
const params = {
|
||||
name: this.secretInfo.name,
|
||||
secret: encryptPassword(this.modifiedSecret)
|
||||
}
|
||||
this.$axios.patch(`/api/v1/accounts/accounts/${this.account.id}/`, params).then(() => {
|
||||
this.$message.success(this.$tc('common.updateSuccessMsg'))
|
||||
})
|
||||
},
|
||||
getAuthInfo() {
|
||||
this.$axios.get(this.url, { disableFlashErrorMsg: true }).then(resp => {
|
||||
this.secretInfo = resp
|
||||
this.sshKeyFingerprint = resp?.spec_info?.ssh_key_fingerprint || '-'
|
||||
this.showSecret = true
|
||||
})
|
||||
},
|
||||
exit() {
|
||||
this.$emit('update:visible', false)
|
||||
},
|
||||
showHistoryDialog() {
|
||||
this.showPasswordHistoryDialog = true
|
||||
},
|
||||
onShowKeyCopyFormatterChange(value) {
|
||||
if (value === this.secretInfo.secret) return
|
||||
this.modifiedSecret = value
|
||||
}
|
||||
}
|
||||
}
|
||||
</script>
|
||||
|
||||
<style lang="scss" scoped>
|
||||
.item-textarea >>> .el-textarea__inner {
|
||||
height: 110px;
|
||||
}
|
||||
|
||||
.el-form-item {
|
||||
border-bottom: 1px solid #EBEEF5;
|
||||
padding: 5px 0;
|
||||
margin-bottom: 0;
|
||||
|
||||
&:last-child {
|
||||
border-bottom: none;
|
||||
}
|
||||
|
||||
>>> .el-form-item__label {
|
||||
padding-right: 20px;
|
||||
line-height: 30px;
|
||||
}
|
||||
|
||||
>>> .el-form-item__content {
|
||||
line-height: 30px;
|
||||
|
||||
pre {
|
||||
margin: 0;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
ul {
|
||||
margin: 0;
|
||||
}
|
||||
|
||||
li {
|
||||
display: block;
|
||||
font-size: 13px;
|
||||
margin-bottom: 8px;
|
||||
white-space: nowrap;
|
||||
overflow: hidden;
|
||||
text-overflow: ellipsis;
|
||||
|
||||
.title {
|
||||
color: #303133;
|
||||
font-weight: 500;
|
||||
}
|
||||
}
|
||||
</style>
|
||||
@@ -1,36 +1,26 @@
|
||||
import { toSafeLocalDateStr } from '@/utils/common'
|
||||
import ChoicesFormatter from '@/components/TableFormatters/ChoicesFormatter'
|
||||
import i18n from '@/i18n/i18n'
|
||||
import { ChoicesFormatter } from '@/components/TableFormatters'
|
||||
|
||||
export const connectivityMeta = {
|
||||
label: i18n.t('assets.Reachable'),
|
||||
label: i18n.t('assets.Connectivity'),
|
||||
formatter: ChoicesFormatter,
|
||||
formatterArgs: {
|
||||
iconChoices: {
|
||||
ok: 'fa-check',
|
||||
failed: 'fa-times',
|
||||
unknown: 'fa-circle-o'
|
||||
faChoices: {
|
||||
'-': '',
|
||||
ok: 'fa-check-circle',
|
||||
err: 'fa-times-circle'
|
||||
},
|
||||
classChoices: {
|
||||
ok: 'text-primary',
|
||||
failed: 'text-danger',
|
||||
unknown: 'text-warning'
|
||||
err: 'text-danger'
|
||||
},
|
||||
hasTips: true,
|
||||
getTips: ({ row, cellValue }) => {
|
||||
const mapper = {
|
||||
'ok': i18n.t('assets.Reachable'),
|
||||
'failed': i18n.t('assets.Unreachable'),
|
||||
'unknown': i18n.t('assets.Unknown')
|
||||
getText({ cellValue }) {
|
||||
if (cellValue?.value === '-' || cellValue?.value === 'unknown') {
|
||||
return '-'
|
||||
} else {
|
||||
return cellValue?.label
|
||||
}
|
||||
let tips = mapper[cellValue]
|
||||
if (row['date_verified']) {
|
||||
const datetime = toSafeLocalDateStr(row['date_verified'])
|
||||
tips += '<br> ' + datetime
|
||||
}
|
||||
return tips
|
||||
}
|
||||
},
|
||||
width: '90px',
|
||||
align: 'center'
|
||||
width: '100px'
|
||||
}
|
||||
|
||||
@@ -1,213 +0,0 @@
|
||||
<template>
|
||||
<div>
|
||||
<ListTable ref="ListTable" :table-config="tableConfig" :header-actions="headerActions" />
|
||||
<ShowSecretInfo v-if="showViewSecretDialog" :visible.sync="showViewSecretDialog" :account="account" />
|
||||
<UpdateSecretInfo v-if="showUpdateSecretDialog" :visible.sync="showUpdateSecretDialog" :account="account" @updateAuthDone="onUpdateAuthDone" />
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<script>
|
||||
import ListTable from '@/components/ListTable/index'
|
||||
import { ActionsFormatter, DetailFormatter, DisplayFormatter } from '@/components/TableFormatters'
|
||||
import ShowSecretInfo from './ShowSecretInfo'
|
||||
import UpdateSecretInfo from './UpdateSecretInfo'
|
||||
import { connectivityMeta } from './const'
|
||||
import { openTaskPage } from '@/utils/jms'
|
||||
// import i18n from '@/i18n/i18n'
|
||||
|
||||
export default {
|
||||
name: 'AccountListTable',
|
||||
components: {
|
||||
ListTable,
|
||||
UpdateSecretInfo,
|
||||
ShowSecretInfo
|
||||
},
|
||||
props: {
|
||||
url: {
|
||||
type: String,
|
||||
required: true
|
||||
},
|
||||
exportUrl: {
|
||||
type: String,
|
||||
default() {
|
||||
return this.url.replace('/assets/accounts/', '/assets/account-secrets/')
|
||||
}
|
||||
},
|
||||
hasLeftActions: {
|
||||
type: Boolean,
|
||||
default: false
|
||||
},
|
||||
otherActions: {
|
||||
type: Array,
|
||||
default: null
|
||||
},
|
||||
hasClone: {
|
||||
type: Boolean,
|
||||
default: false
|
||||
}
|
||||
},
|
||||
data() {
|
||||
const vm = this
|
||||
return {
|
||||
showViewSecretDialog: false,
|
||||
showUpdateSecretDialog: false,
|
||||
account: {},
|
||||
tableConfig: {
|
||||
url: this.url,
|
||||
permissions: {
|
||||
app: 'assets',
|
||||
resource: 'authbook'
|
||||
},
|
||||
columns: [
|
||||
'hostname', 'ip', 'username', 'version', 'connectivity',
|
||||
'systemuser', 'date_created', 'date_updated', 'actions'
|
||||
],
|
||||
columnsShow: {
|
||||
min: ['username', 'actions'],
|
||||
default: ['hostname', 'ip', 'username', 'version', 'actions']
|
||||
},
|
||||
columnsMeta: {
|
||||
hostname: {
|
||||
prop: 'hostname',
|
||||
label: this.$t('assets.Hostname'),
|
||||
showOverflowTooltip: true,
|
||||
formatter: DetailFormatter,
|
||||
formatterArgs: {
|
||||
can: this.$hasPerm('assets.view_asset'),
|
||||
getRoute({ row }) {
|
||||
return {
|
||||
name: 'AssetDetail',
|
||||
params: { id: row.asset }
|
||||
}
|
||||
}
|
||||
}
|
||||
},
|
||||
ip: {
|
||||
width: '120px'
|
||||
},
|
||||
username: {
|
||||
showOverflowTooltip: true
|
||||
},
|
||||
systemuser: {
|
||||
formatter: DisplayFormatter
|
||||
},
|
||||
version: {
|
||||
width: '70px'
|
||||
},
|
||||
connectivity: connectivityMeta,
|
||||
actions: {
|
||||
formatter: ActionsFormatter,
|
||||
formatterArgs: {
|
||||
hasUpdate: false, // can set function(row, value)
|
||||
hasDelete: false, // can set function(row, value)
|
||||
hasClone: this.hasClone,
|
||||
moreActionsTitle: this.$t('common.More'),
|
||||
extraActions: [
|
||||
{
|
||||
name: 'View',
|
||||
title: this.$t('common.View'),
|
||||
can: this.$hasPerm('assets.view_assetaccountsecret'),
|
||||
type: 'primary',
|
||||
callback: ({ row }) => {
|
||||
vm.account = row
|
||||
vm.showViewSecretDialog = false
|
||||
setTimeout(() => {
|
||||
vm.showViewSecretDialog = true
|
||||
})
|
||||
}
|
||||
},
|
||||
{
|
||||
name: 'Delete',
|
||||
title: this.$t('common.Delete'),
|
||||
can: this.$hasPerm('assets.delete_authbook'),
|
||||
type: 'primary',
|
||||
callback: ({ row }) => {
|
||||
this.$axios.delete(`/api/v1/assets/accounts/${row.id}/`).then(() => {
|
||||
this.$message.success(this.$tc('common.deleteSuccessMsg'))
|
||||
this.$refs.ListTable.reloadTable()
|
||||
})
|
||||
}
|
||||
},
|
||||
{
|
||||
name: 'Test',
|
||||
title: this.$t('common.Test'),
|
||||
can: this.$hasPerm('assets.test_authbook'),
|
||||
callback: ({ row }) => {
|
||||
this.$axios.post(
|
||||
`/api/v1/assets/accounts/${row.id}/verify/`,
|
||||
{ action: 'test' }
|
||||
).then(res => {
|
||||
openTaskPage(res['task'])
|
||||
})
|
||||
}
|
||||
},
|
||||
{
|
||||
name: 'Update',
|
||||
title: this.$t('common.Update'),
|
||||
can: this.$hasPerm('assets.change_assetaccountsecret') && !this.$store.getters.currentOrgIsRoot,
|
||||
callback: ({ row }) => {
|
||||
vm.account = row
|
||||
vm.showUpdateSecretDialog = false
|
||||
setTimeout(() => {
|
||||
vm.showUpdateSecretDialog = true
|
||||
})
|
||||
}
|
||||
}
|
||||
// {
|
||||
// name: 'History',
|
||||
// title: i18n.t('common.History'),
|
||||
// can: this.$hasPerm('assets.view_assethistoryaccount') && !this.$store.getters.currentOrgIsRoot,
|
||||
// callback: ({ row }) => {
|
||||
// this.$router.push({
|
||||
// name: 'AssetAccountHistoryList',
|
||||
// query: { id: row.id }
|
||||
// })
|
||||
// }
|
||||
// }
|
||||
]
|
||||
}
|
||||
}
|
||||
}
|
||||
},
|
||||
headerActions: {
|
||||
hasLeftActions: this.hasLeftActions,
|
||||
hasMoreActions: true,
|
||||
hasCreate: false,
|
||||
hasImport: false,
|
||||
hasExport: this.$hasPerm('assets.view_assetaccountsecret'),
|
||||
exportOptions: {
|
||||
url: this.exportUrl,
|
||||
mfaVerifyRequired: true
|
||||
},
|
||||
searchConfig: {
|
||||
exclude: ['systemuser', 'asset']
|
||||
},
|
||||
hasSearch: true
|
||||
}
|
||||
}
|
||||
},
|
||||
watch: {
|
||||
url(iNew) {
|
||||
this.$set(this.tableConfig, 'url', iNew)
|
||||
this.$set(this.headerActions.exportOptions, 'url', iNew.replace('/accounts/', '/account-secrets/'))
|
||||
}
|
||||
},
|
||||
mounted() {
|
||||
if (this.otherActions) {
|
||||
const actionColumn = this.tableConfig.columns[this.tableConfig.columns.length - 1]
|
||||
for (const item of this.otherActions) {
|
||||
actionColumn.formatterArgs.extraActions.push(item)
|
||||
}
|
||||
}
|
||||
},
|
||||
methods: {
|
||||
onUpdateAuthDone(account) {
|
||||
Object.assign(this.account, account)
|
||||
}
|
||||
}
|
||||
}
|
||||
</script>
|
||||
|
||||
<style lang='less' scoped>
|
||||
|
||||
</style>
|
||||
@@ -18,16 +18,16 @@ export default {
|
||||
type: Array,
|
||||
default: () => []
|
||||
},
|
||||
moreActionBtn: {
|
||||
type: Object,
|
||||
default: () => ({})
|
||||
},
|
||||
moreActionsTitle: {
|
||||
type: String,
|
||||
default() {
|
||||
return this.$t('common.MoreActions')
|
||||
}
|
||||
},
|
||||
moreActionsType: {
|
||||
type: String,
|
||||
default: 'default'
|
||||
},
|
||||
moreActionsPlacement: {
|
||||
type: String,
|
||||
default: 'bottom'
|
||||
@@ -43,14 +43,21 @@ export default {
|
||||
return actions
|
||||
},
|
||||
iMoreAction() {
|
||||
return {
|
||||
const defaultBtn = {
|
||||
name: 'moreActions',
|
||||
title: this.iMoreActionsTitle,
|
||||
title: this.$t('common.MoreActions'),
|
||||
type: 'primary',
|
||||
plain: true
|
||||
}
|
||||
const btn = {
|
||||
...defaultBtn,
|
||||
...this.moreActionBtn,
|
||||
dropdown: this.moreActions || []
|
||||
}
|
||||
},
|
||||
iMoreActionsTitle() {
|
||||
return this.moreActionsTitle || this.$t('common.MoreActions')
|
||||
if (this.moreActionsTitle) {
|
||||
btn.title = this.moreActionsTitle
|
||||
}
|
||||
return btn
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -9,10 +9,10 @@
|
||||
>
|
||||
<span class="announcement-main">{{ announcement.content }}</span>
|
||||
<span v-if="announcement.link">
|
||||
<el-link :href="announcement.link" target="_blank" class="link-more">
|
||||
<el-link :href="announcement.link" target="_blank" type="info" class="link-more">
|
||||
{{ $t('common.ViewMore') }}
|
||||
</el-link>
|
||||
<i class="fa fa-external-link" />
|
||||
<i class="fa fa-external-link icon" />
|
||||
</span>
|
||||
</el-alert>
|
||||
</template>
|
||||
@@ -67,5 +67,7 @@ export default {
|
||||
margin-left: 10px;
|
||||
border-bottom: solid 1px;
|
||||
}
|
||||
|
||||
.icon {
|
||||
vertical-align: text-bottom;
|
||||
}
|
||||
</style>
|
||||
|
||||
@@ -1,78 +0,0 @@
|
||||
<template>
|
||||
<div>
|
||||
<UserConfirmDialog
|
||||
:url="url"
|
||||
@UserConfirmDone="getAuthInfo"
|
||||
@UserConfirmCancel="exit"
|
||||
/>
|
||||
<Dialog
|
||||
:title="dialogTitle"
|
||||
:show-confirm="false"
|
||||
:show-cancel="false"
|
||||
:destroy-on-close="true"
|
||||
:width="'50'"
|
||||
:visible.sync="showAuthInfo"
|
||||
v-bind="$attrs"
|
||||
v-on="$listeners"
|
||||
>
|
||||
<div>
|
||||
<el-form label-position="right" label-width="80px" :model="authInfo">
|
||||
<el-form-item :label="this.$t('applications.appName')">
|
||||
<el-input v-model="account['app_display']" readonly />
|
||||
</el-form-item>
|
||||
<el-form-item :label="this.$t('assets.Username')">
|
||||
<el-input v-model="account['username']" readonly />
|
||||
</el-form-item>
|
||||
<el-form-item :label="this.$t('assets.Password')">
|
||||
<el-input v-model="authInfo.password" type="password" show-password />
|
||||
</el-form-item>
|
||||
</el-form>
|
||||
</div>
|
||||
</Dialog>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<script>
|
||||
import Dialog from '@/components/Dialog'
|
||||
import UserConfirmDialog from '@/components/UserConfirmDialog'
|
||||
export default {
|
||||
name: 'ShowSecretInfo',
|
||||
components: {
|
||||
Dialog,
|
||||
UserConfirmDialog
|
||||
},
|
||||
props: {
|
||||
account: {
|
||||
type: Object,
|
||||
default: () => ({})
|
||||
},
|
||||
visible: {
|
||||
type: Boolean,
|
||||
default: false
|
||||
}
|
||||
},
|
||||
data() {
|
||||
return {
|
||||
dialogTitle: this.$t('common.ViewSecret'),
|
||||
authInfo: {},
|
||||
showAuthInfo: false,
|
||||
url: `/api/v1/applications/account-secrets/${this.account.id}/`
|
||||
}
|
||||
},
|
||||
methods: {
|
||||
getAuthInfo() {
|
||||
this.$axios.get(this.url, { disableFlashErrorMsg: true }).then(resp => {
|
||||
this.authInfo = resp
|
||||
this.showAuthInfo = true
|
||||
})
|
||||
},
|
||||
exit() {
|
||||
this.$emit('update:visible', false)
|
||||
}
|
||||
}
|
||||
}
|
||||
</script>
|
||||
|
||||
<style scoped>
|
||||
|
||||
</style>
|
||||
@@ -1,31 +0,0 @@
|
||||
import { ChoicesFormatter } from '@/components/TableFormatters'
|
||||
import { toSafeLocalDateStr } from '@/utils/common'
|
||||
import i18n from '@/i18n/i18n'
|
||||
|
||||
export const connectivityMeta = {
|
||||
label: i18n.t('assets.Reachable'),
|
||||
formatter: ChoicesFormatter,
|
||||
formatterArgs: {
|
||||
iconChoices: {
|
||||
ok: 'fa-check text-primary',
|
||||
failed: 'fa-times text-danger',
|
||||
unknown: 'fa-circle text-warning'
|
||||
},
|
||||
hasTips: true,
|
||||
getTips: ({ row, cellValue }) => {
|
||||
const mapper = {
|
||||
'ok': i18n.t('assets.Reachable'),
|
||||
'failed': i18n.t('assets.Unreachable'),
|
||||
'unknown': i18n.t('assets.Unknown')
|
||||
}
|
||||
let tips = mapper[cellValue]
|
||||
if (row['date_verified']) {
|
||||
const datetime = toSafeLocalDateStr(row['date_verified'])
|
||||
tips += '<br> ' + datetime
|
||||
}
|
||||
return tips
|
||||
}
|
||||
},
|
||||
width: '90px',
|
||||
align: 'center'
|
||||
}
|
||||
@@ -1,173 +0,0 @@
|
||||
<template>
|
||||
<div>
|
||||
<ListTable ref="ListTable" :table-config="tableConfig" :header-actions="headerActions" />
|
||||
<ShowSecretInfo v-if="showViewSecretDialog" :visible.sync="showViewSecretDialog" :account="account" />
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<script>
|
||||
import ListTable from '@/components/ListTable/index'
|
||||
import { ActionsFormatter, DetailFormatter } from '@/components/TableFormatters'
|
||||
import ShowSecretInfo from './ShowSecretInfo'
|
||||
|
||||
export default {
|
||||
name: 'Detail',
|
||||
components: {
|
||||
ListTable,
|
||||
ShowSecretInfo
|
||||
},
|
||||
props: {
|
||||
url: {
|
||||
type: String,
|
||||
required: true
|
||||
},
|
||||
exportUrl: {
|
||||
type: String,
|
||||
default() {
|
||||
return this.url.replace('/applications/accounts/', '/applications/account-secrets/')
|
||||
}
|
||||
},
|
||||
hasLeftActions: {
|
||||
type: Boolean,
|
||||
default: false
|
||||
},
|
||||
otherActions: {
|
||||
type: Array,
|
||||
default: null
|
||||
},
|
||||
hasClone: {
|
||||
type: Boolean,
|
||||
default: false
|
||||
},
|
||||
systemUserDisabled: {
|
||||
type: Boolean,
|
||||
default: true
|
||||
}
|
||||
},
|
||||
data() {
|
||||
return {
|
||||
showViewSecretDialog: false,
|
||||
showUpdateSecretDialog: false,
|
||||
account: {},
|
||||
tableConfig: {
|
||||
url: this.url,
|
||||
columns: [
|
||||
'app_display', 'username', 'category_display',
|
||||
'type_display', 'systemuser', 'actions'
|
||||
],
|
||||
columnsMeta: {
|
||||
app_display: {
|
||||
showOverflowTooltip: true,
|
||||
formatter: DetailFormatter,
|
||||
formatterArgs: {
|
||||
getRoute({ row }) {
|
||||
switch (row['category']) {
|
||||
case 'remote_app':
|
||||
return {
|
||||
name: 'RemoteAppDetail',
|
||||
params: { id: row.app }
|
||||
}
|
||||
case 'db':
|
||||
return {
|
||||
name: 'DatabaseAppDetail',
|
||||
params: { id: row.app }
|
||||
}
|
||||
default:
|
||||
return {
|
||||
name: 'KubernetesAppDetail',
|
||||
params: { id: row.app }
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
},
|
||||
username: {
|
||||
showOverflowTooltip: true
|
||||
},
|
||||
systemuser: {
|
||||
showOverflowTooltip: true,
|
||||
formatter: DetailFormatter,
|
||||
formatterArgs: {
|
||||
can: this.systemUserDisabled && this.$hasPerm('assets.view_systemuser'),
|
||||
getTitle({ row }) {
|
||||
return row.systemuser_display
|
||||
},
|
||||
getRoute({ row }) {
|
||||
return {
|
||||
name: 'SystemUserDetail',
|
||||
params: { id: row.systemuser }
|
||||
}
|
||||
}
|
||||
}
|
||||
},
|
||||
actions: {
|
||||
formatter: ActionsFormatter,
|
||||
formatterArgs: {
|
||||
hasUpdate: false, // can set function(row, value)
|
||||
hasDelete: false, // can set function(row, value)
|
||||
hasClone: this.hasClone,
|
||||
moreActionsTitle: this.$t('common.More'),
|
||||
extraActions: [
|
||||
{
|
||||
name: 'View',
|
||||
title: this.$t('common.View'),
|
||||
type: 'primary',
|
||||
can: this.$hasPerm('applications.view_applicationaccountsecret'),
|
||||
callback: function({ row }) {
|
||||
this.account = row
|
||||
this.showViewSecretDialog = true
|
||||
}.bind(this)
|
||||
},
|
||||
{
|
||||
name: 'Update',
|
||||
title: this.$t('common.Update'),
|
||||
can: !this.$store.getters.currentOrgIsRoot,
|
||||
callback: function({ row }) {
|
||||
this.$message.success(this.$tc('applications.updateAccountMsg'))
|
||||
}.bind(this)
|
||||
}
|
||||
]
|
||||
}
|
||||
}
|
||||
}
|
||||
},
|
||||
headerActions: {
|
||||
hasLeftActions: this.hasLeftActions,
|
||||
hasMoreActions: false,
|
||||
hasImport: false,
|
||||
hasExport: this.$hasPerm('applications.view_applicationaccountsecret'),
|
||||
exportOptions: {
|
||||
url: this.exportUrl,
|
||||
mfaVerifyRequired: true
|
||||
},
|
||||
searchConfig: {
|
||||
exclude: ['systemuser', 'app']
|
||||
},
|
||||
hasSearch: true
|
||||
}
|
||||
}
|
||||
},
|
||||
watch: {
|
||||
url(iNew) {
|
||||
this.$set(this.tableConfig, 'url', iNew)
|
||||
}
|
||||
},
|
||||
mounted() {
|
||||
if (this.otherActions) {
|
||||
const actionColumn = this.tableConfig.columns[this.tableConfig.columns.length - 1]
|
||||
for (const item of this.otherActions) {
|
||||
actionColumn.formatterArgs.extraActions.push(item)
|
||||
}
|
||||
}
|
||||
},
|
||||
methods: {
|
||||
onUpdateAuthDone(account) {
|
||||
Object.assign(this.account, account)
|
||||
}
|
||||
}
|
||||
}
|
||||
</script>
|
||||
|
||||
<style lang='less' scoped>
|
||||
|
||||
</style>
|
||||
192
src/components/AssetSelect/dialog.vue
Normal file
192
src/components/AssetSelect/dialog.vue
Normal file
@@ -0,0 +1,192 @@
|
||||
<template>
|
||||
<Dialog
|
||||
:title="$tc('assets.Assets')"
|
||||
custom-class="asset-select-dialog"
|
||||
top="1vh"
|
||||
v-bind="$attrs"
|
||||
width="80vw"
|
||||
@cancel="handleCancel"
|
||||
@close="handleClose"
|
||||
@confirm="handleConfirm"
|
||||
v-on="$listeners"
|
||||
>
|
||||
<AssetTreeTable
|
||||
ref="ListPage"
|
||||
:header-actions="headerActions"
|
||||
:node-url="baseNodeUrl"
|
||||
:table-config="tableConfig"
|
||||
:tree-url="`${baseNodeUrl}children/tree/`"
|
||||
:url="baseUrl"
|
||||
class="tree-table"
|
||||
v-bind="$attrs"
|
||||
/>
|
||||
</Dialog>
|
||||
</template>
|
||||
|
||||
<script>
|
||||
import AssetTreeTable from '@/components/AssetTreeTable'
|
||||
import Dialog from '@/components/Dialog'
|
||||
|
||||
export default {
|
||||
componentName: 'AssetSelectDialog',
|
||||
components: { AssetTreeTable, Dialog },
|
||||
props: {
|
||||
baseUrl: {
|
||||
type: String,
|
||||
default: '/api/v1/assets/assets/'
|
||||
},
|
||||
baseNodeUrl: {
|
||||
type: String,
|
||||
default: '/api/v1/assets/nodes/'
|
||||
},
|
||||
value: {
|
||||
type: Array,
|
||||
default: () => []
|
||||
},
|
||||
canSelect: {
|
||||
type: Function,
|
||||
default(row, index) {
|
||||
return true
|
||||
}
|
||||
},
|
||||
disabled: {
|
||||
type: [Boolean, Function],
|
||||
default: false
|
||||
}
|
||||
},
|
||||
data() {
|
||||
const vm = this
|
||||
return {
|
||||
dialogVisible: false,
|
||||
rowSelected: _.cloneDeep(this.value) || [],
|
||||
rowsAdd: [],
|
||||
tableConfig: {
|
||||
url: this.baseUrl,
|
||||
hasTree: true,
|
||||
canSelect: this.canSelect,
|
||||
columns: [
|
||||
{
|
||||
prop: 'name',
|
||||
label: this.$t('assets.Name'),
|
||||
sortable: true
|
||||
},
|
||||
{
|
||||
prop: 'address',
|
||||
label: this.$t('assets.ipDomain'),
|
||||
sortable: 'custom'
|
||||
},
|
||||
{
|
||||
prop: 'platform',
|
||||
label: this.$t('assets.Platform'),
|
||||
sortable: true,
|
||||
formatter: function(row) {
|
||||
return row.platform.name
|
||||
}
|
||||
},
|
||||
{
|
||||
prop: 'protocols',
|
||||
formatter: function(row) {
|
||||
const data = row.protocols.map(p => {
|
||||
return <el-tag size='mini'>{p.name}/{p.port} </el-tag>
|
||||
})
|
||||
return <span> {data} </span>
|
||||
},
|
||||
label: this.$t('assets.Protocols')
|
||||
},
|
||||
{
|
||||
prop: 'actions',
|
||||
has: false
|
||||
}
|
||||
],
|
||||
listeners: {
|
||||
'toggle-row-selection': (isSelected, row) => {
|
||||
if (isSelected) {
|
||||
vm.addRowToSelect(row)
|
||||
} else {
|
||||
vm.removeRowFromSelect(row)
|
||||
}
|
||||
}
|
||||
},
|
||||
theRowDefaultIsSelected: (row) => {
|
||||
return this.value.indexOf(row.id) > -1
|
||||
}
|
||||
},
|
||||
headerActions: {
|
||||
hasLeftActions: false,
|
||||
hasRightActions: false,
|
||||
searchConfig: {
|
||||
getUrlQuery: false
|
||||
}
|
||||
}
|
||||
}
|
||||
},
|
||||
methods: {
|
||||
handleClose() {
|
||||
this.$eventBus.$emit('treeComponentKey')
|
||||
},
|
||||
handleConfirm() {
|
||||
this.$emit('confirm', this.rowSelected, this.rowsAdd)
|
||||
if (this.rowSelected.length > 0) {
|
||||
this.handleClose()
|
||||
}
|
||||
},
|
||||
handleCancel() {
|
||||
this.$emit('cancel')
|
||||
this.handleClose()
|
||||
},
|
||||
addRowToSelect(row) {
|
||||
const selectValueIndex = this.rowSelected.indexOf(row.id)
|
||||
if (selectValueIndex === -1) {
|
||||
this.rowSelected.push(row.id)
|
||||
this.rowsAdd.push(row)
|
||||
}
|
||||
},
|
||||
removeRowFromSelect(row) {
|
||||
const selectValueIndex = this.rowSelected.indexOf(row.id)
|
||||
if (selectValueIndex > -1) {
|
||||
this.rowSelected.splice(selectValueIndex, 1)
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
</script>
|
||||
|
||||
<style lang="scss" scoped>
|
||||
.page ::v-deep .page-heading {
|
||||
display: none;
|
||||
}
|
||||
|
||||
.el-dialog__wrapper ::v-deep .el-dialog__body {
|
||||
padding: 0 0 0 3px;
|
||||
|
||||
.tree-table {
|
||||
.search {
|
||||
}
|
||||
|
||||
.left {
|
||||
padding: 5px;
|
||||
}
|
||||
|
||||
.right {
|
||||
height: calc(100vh - 200px);
|
||||
overflow: auto;
|
||||
}
|
||||
|
||||
.mini {
|
||||
padding-top: 8px;
|
||||
}
|
||||
|
||||
.transition-box {
|
||||
padding: 10px 5px;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
.page ::v-deep .treebox .ztree {
|
||||
|
||||
}
|
||||
|
||||
.asset-select-dialog ::v-deep .el-icon-circle-check {
|
||||
display: none;
|
||||
}
|
||||
</style>
|
||||
@@ -1,151 +1,96 @@
|
||||
<template>
|
||||
<div class="asset-select-dialog">
|
||||
<div class="asset-select-formatter">
|
||||
<Select2
|
||||
ref="select2"
|
||||
v-model="select2Config.value"
|
||||
v-bind="select2Config"
|
||||
@input="onInputChange"
|
||||
v-on="$listeners"
|
||||
@focus.stop="handleFocus"
|
||||
/>
|
||||
<AssetSelectDialog
|
||||
v-if="dialogVisible"
|
||||
ref="dialog"
|
||||
:base-node-url="baseNodeUrl"
|
||||
:base-url="baseUrl"
|
||||
:tree-url-query="treeUrlQuery"
|
||||
:value="value"
|
||||
:visible.sync="dialogVisible"
|
||||
v-bind="$attrs"
|
||||
@cancel="handleCancel"
|
||||
@confirm="handleConfirm"
|
||||
v-on="$listeners"
|
||||
/>
|
||||
<Dialog
|
||||
v-if="dialogVisible"
|
||||
:title="this.$t('assets.Assets')"
|
||||
:visible.sync="dialogVisible"
|
||||
custom-class="asset-select-dialog"
|
||||
width="80vw"
|
||||
top="1vh"
|
||||
@confirm="handleConfirm"
|
||||
@cancel="handleCancel"
|
||||
>
|
||||
<TreeTable
|
||||
ref="ListPage"
|
||||
:tree-setting="treeSetting"
|
||||
:table-config="tableConfig"
|
||||
:header-actions="headerActions"
|
||||
/>
|
||||
</Dialog>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<script>
|
||||
import TreeTable from '@/components/TreeTable'
|
||||
import { DetailFormatter } from '@/components/TableFormatters'
|
||||
import Select2 from '@/components/FormFields/Select2'
|
||||
import Dialog from '@/components/Dialog'
|
||||
import AssetSelectDialog from './dialog.vue'
|
||||
import { b } from 'css-color-function/lib/adjusters'
|
||||
|
||||
export default {
|
||||
componentName: 'AssetSelect',
|
||||
components: { TreeTable, Select2, Dialog },
|
||||
components: { AssetSelectDialog, Select2 },
|
||||
props: {
|
||||
baseUrl: {
|
||||
type: String,
|
||||
default: '/api/v1/assets/assets/'
|
||||
},
|
||||
baseNodeUrl: {
|
||||
type: String,
|
||||
default: '/api/v1/assets/nodes/'
|
||||
},
|
||||
treeUrlQuery: {
|
||||
type: Object,
|
||||
default: () => {}
|
||||
},
|
||||
value: {
|
||||
type: Array,
|
||||
default: () => []
|
||||
},
|
||||
canSelect: {
|
||||
type: Function,
|
||||
default(row, index) {
|
||||
return true
|
||||
}
|
||||
},
|
||||
disabled: {
|
||||
type: [Boolean, Function],
|
||||
default: false
|
||||
}
|
||||
},
|
||||
data() {
|
||||
const select2Config = {
|
||||
value: this.value,
|
||||
multiple: true,
|
||||
clearable: true,
|
||||
ajax: {
|
||||
url: '/api/v1/assets/assets/?fields_size=mini',
|
||||
transformOption: (item) => {
|
||||
return { label: item.hostname + '(' + item.ip + ')', value: item.id }
|
||||
}
|
||||
const iValue = []
|
||||
for (const item of this.value) {
|
||||
if (typeof item === 'object') {
|
||||
iValue.push(item.id)
|
||||
} else {
|
||||
iValue.push(item)
|
||||
}
|
||||
}
|
||||
const vm = this
|
||||
return {
|
||||
dialogVisible: false,
|
||||
initialValue: _.cloneDeep(this.value),
|
||||
rowSelected: [],
|
||||
initSelection: null,
|
||||
treeSetting: {
|
||||
showMenu: false,
|
||||
showRefresh: true,
|
||||
showAssets: false,
|
||||
showSearch: true,
|
||||
customTreeHeader: true,
|
||||
url: '/api/v1/assets/assets/?fields_size=mini',
|
||||
nodeUrl: '/api/v1/assets/nodes/',
|
||||
// ?assets=0不显示资产. =1显示资产
|
||||
treeUrl: '/api/v1/assets/nodes/children/tree/?assets=0'
|
||||
},
|
||||
select2Config: select2Config,
|
||||
dialogSelect2Config: select2Config,
|
||||
tableConfig: {
|
||||
url: '/api/v1/assets/assets/?fields_size=mini',
|
||||
hasTree: true,
|
||||
canSelect: this.canSelect,
|
||||
columns: [
|
||||
{
|
||||
prop: 'hostname',
|
||||
label: this.$t('assets.Hostname'),
|
||||
sortable: true,
|
||||
showOverflowTooltip: true,
|
||||
formatter: DetailFormatter,
|
||||
formatterArgs: {
|
||||
route: 'AssetDetail'
|
||||
}
|
||||
},
|
||||
{
|
||||
prop: 'ip',
|
||||
label: this.$t('assets.ipDomain'),
|
||||
sortable: 'custom'
|
||||
},
|
||||
{
|
||||
prop: 'platform',
|
||||
label: this.$t('assets.Platform'),
|
||||
sortable: true
|
||||
},
|
||||
{
|
||||
prop: 'protocols',
|
||||
formatter: function(row) {
|
||||
return <span> {row.protocols.toString()} </span>
|
||||
},
|
||||
label: this.$t('assets.Protocols')
|
||||
initialValue: _.cloneDeep(iValue),
|
||||
select2Config: {
|
||||
value: iValue,
|
||||
multiple: true,
|
||||
clearable: true,
|
||||
ajax: {
|
||||
url: this.baseUrl,
|
||||
transformOption: (item) => {
|
||||
return { label: item.name + '(' + item.address + ')', value: item.id }
|
||||
}
|
||||
],
|
||||
listeners: {
|
||||
'toggle-row-selection': (isSelected, row) => {
|
||||
if (isSelected) {
|
||||
vm.addRowToSelect(row)
|
||||
} else {
|
||||
vm.removeRowFromSelect(row)
|
||||
}
|
||||
}
|
||||
},
|
||||
theRowDefaultIsSelected: (row) => {
|
||||
return this.value.indexOf(row.id) > -1
|
||||
}
|
||||
},
|
||||
headerActions: {
|
||||
hasLeftActions: false,
|
||||
hasRightActions: false
|
||||
}
|
||||
}
|
||||
},
|
||||
methods: {
|
||||
b,
|
||||
handleFocus() {
|
||||
this.$refs.select2.selectRef.blur()
|
||||
this.dialogVisible = true
|
||||
},
|
||||
handleConfirm() {
|
||||
handleConfirm(valueSelected, rowsAdd) {
|
||||
if (valueSelected === undefined) {
|
||||
return
|
||||
}
|
||||
this.$refs.select2.iValue = valueSelected
|
||||
this.addRowsToSelect(rowsAdd)
|
||||
this.onInputChange(valueSelected)
|
||||
this.dialogVisible = false
|
||||
},
|
||||
handleCancel() {
|
||||
this.$refs.select2.iValue = this.initialValue
|
||||
this.dialogVisible = false
|
||||
},
|
||||
onInputChange(val) {
|
||||
@@ -156,53 +101,54 @@ export default {
|
||||
// 如果select2的options中没有,那么可能无法显示正常的值
|
||||
if (selectOptionsHas === undefined) {
|
||||
const option = {
|
||||
label: `${row.hostname}(${row.ip})`,
|
||||
label: `${row.name}(${row.address})`,
|
||||
value: row.id
|
||||
}
|
||||
options.push(option)
|
||||
}
|
||||
},
|
||||
addRowToSelect(row) {
|
||||
addRowsToSelect(rows) {
|
||||
const outSelectOptions = this.$refs.select2.options
|
||||
this.addToSelect(outSelectOptions, row)
|
||||
|
||||
const selectValue = this.$refs.select2.iValue
|
||||
const selectValueIndex = selectValue.indexOf(row.id)
|
||||
if (selectValueIndex === -1) {
|
||||
selectValue.push(row.id)
|
||||
}
|
||||
this.onInputChange(selectValue)
|
||||
},
|
||||
removeRowFromSelect(row) {
|
||||
const selectValue = this.$refs.select2.iValue
|
||||
const selectValueIndex = selectValue.indexOf(row.id)
|
||||
if (selectValueIndex > -1) {
|
||||
selectValue.splice(selectValueIndex, 1)
|
||||
for (const row of rows) {
|
||||
this.addToSelect(outSelectOptions, row)
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
</script>
|
||||
|
||||
<style lang='scss' scoped>
|
||||
.el-select{
|
||||
width: 100%;
|
||||
}
|
||||
.page ::v-deep .page-heading{
|
||||
display: none;
|
||||
}
|
||||
.el-dialog__wrapper ::v-deep .el-dialog__body{
|
||||
padding: 5px 10px;
|
||||
}
|
||||
.page ::v-deep .treebox {
|
||||
height: inherit !important;
|
||||
}
|
||||
.asset-select-dialog >>> .transition-box:first-child {
|
||||
background-color: #f3f3f3 ;
|
||||
}
|
||||
<style lang="scss" scoped>
|
||||
.el-select {
|
||||
width: 100%;
|
||||
}
|
||||
|
||||
.el-dialog__wrapper ::v-deep .el-dialog__body .wrapper-content {
|
||||
padding: 10px;
|
||||
}
|
||||
.page ::v-deep .page-heading {
|
||||
display: none;
|
||||
}
|
||||
|
||||
.el-dialog__wrapper ::v-deep .el-dialog__body {
|
||||
padding: 0 0 0 3px;
|
||||
|
||||
.tree-table {
|
||||
.left {
|
||||
padding: 5px;
|
||||
|
||||
.ztree {
|
||||
height: calc(100vh - 250px) !important;
|
||||
}
|
||||
}
|
||||
|
||||
.mini {
|
||||
padding-top: 8px;
|
||||
}
|
||||
|
||||
.transition-box {
|
||||
padding: 10px 5px;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
.page ::v-deep .treebox {
|
||||
height: inherit !important;
|
||||
}
|
||||
</style>
|
||||
|
||||
175
src/components/AssetTreeTable/index.vue
Normal file
175
src/components/AssetTreeTable/index.vue
Normal file
@@ -0,0 +1,175 @@
|
||||
<template>
|
||||
<TreeTable
|
||||
ref="TreeList"
|
||||
component="TabTree"
|
||||
:table-config="tableConfig"
|
||||
:active-menu.sync="treeTableConfig.activeMenu"
|
||||
:tree-tab-config="treeTableConfig"
|
||||
v-bind="$attrs"
|
||||
v-on="$listeners"
|
||||
>
|
||||
<template #table>
|
||||
<slot name="table" />
|
||||
</template>
|
||||
<div slot="rMenu" slot-scope="{data}">
|
||||
<slot name="rMenu" :data="data" />
|
||||
</div>
|
||||
</TreeTable>
|
||||
</template>
|
||||
|
||||
<script>
|
||||
import TreeTable from '../TreeTable'
|
||||
import { setRouterQuery, setUrlParam } from '@/utils/common'
|
||||
import $ from '@/utils/jquery-vendor'
|
||||
|
||||
export default {
|
||||
components: {
|
||||
TreeTable
|
||||
},
|
||||
props: {
|
||||
url: {
|
||||
type: String,
|
||||
default: '/api/v1/assets/assets/'
|
||||
},
|
||||
nodeUrl: {
|
||||
type: String,
|
||||
default: '/api/v1/assets/nodes/'
|
||||
},
|
||||
treeUrl: {
|
||||
type: String,
|
||||
default: '/api/v1/assets/nodes/children/tree/'
|
||||
},
|
||||
treeUrlQuery: {
|
||||
type: Object,
|
||||
default: () => ({})
|
||||
},
|
||||
treeSetting: {
|
||||
type: Object,
|
||||
default: () => ({})
|
||||
},
|
||||
tableConfig: {
|
||||
type: Object,
|
||||
default: () => ({})
|
||||
},
|
||||
showAssets: {
|
||||
type: Boolean,
|
||||
default: false
|
||||
}
|
||||
},
|
||||
data() {
|
||||
const showAssets = this.treeSetting?.showAssets || this.showAssets
|
||||
const treeUrlQuery = this.setTreeUrlQuery()
|
||||
const assetTreeUrl = `${this.treeUrl}?assets=${showAssets ? '1' : '0'}&${treeUrlQuery}`
|
||||
|
||||
return {
|
||||
treeTabConfig: {
|
||||
activeMenu: 'CustomTree',
|
||||
submenu: [
|
||||
{
|
||||
title: this.$t('assets.AssetTree'),
|
||||
name: 'CustomTree',
|
||||
treeSetting: {
|
||||
showAssets,
|
||||
showMenu: false,
|
||||
showRefresh: true,
|
||||
showCreate: true,
|
||||
showUpdate: true,
|
||||
showDelete: true,
|
||||
hasRightMenu: true,
|
||||
showSearch: true,
|
||||
url: this.url,
|
||||
nodeUrl: this.nodeUrl,
|
||||
treeUrl: assetTreeUrl,
|
||||
callback: {
|
||||
onSelected: (event, treeNode) => this.getAssetsUrl(treeNode)
|
||||
},
|
||||
...this.treeSetting
|
||||
}
|
||||
},
|
||||
{
|
||||
title: this.$t('assets.BuiltinTree'),
|
||||
name: 'BuiltinTree',
|
||||
treeSetting: {
|
||||
showRefresh: true,
|
||||
showAssets: false,
|
||||
showSearch: false,
|
||||
customTreeHeaderName: this.$t('assets.BuiltinTree'),
|
||||
url: '/api/v1/assets/nodes/category/tree/',
|
||||
nodeUrl: this.treeSetting?.nodeUrl || this.nodeUrl,
|
||||
treeUrl: `/api/v1/assets/nodes/category/tree/?assets=${showAssets ? '1' : '0'}&count_resource=${this.treeSetting.countResource || 'asset'}`,
|
||||
callback: {
|
||||
onSelected: (event, treeNode) => this.getAssetsUrl(treeNode)
|
||||
}
|
||||
}
|
||||
}
|
||||
]
|
||||
}
|
||||
}
|
||||
},
|
||||
computed: {
|
||||
treeTableConfig() {
|
||||
if (this.treeSetting.notShowBuiltinTree) {
|
||||
// eslint-disable-next-line vue/no-side-effects-in-computed-properties
|
||||
this.treeTabConfig.submenu.splice(1, 1)
|
||||
}
|
||||
return this.treeTabConfig
|
||||
}
|
||||
},
|
||||
mounted() {
|
||||
this.decorateRMenu()
|
||||
const treeSetting = this.treeTabConfig.submenu[0].treeSetting
|
||||
treeSetting.hasRightMenu = !this.currentOrgIsRoot
|
||||
treeSetting.showCreate = this.$hasPerm('assets.add_node')
|
||||
treeSetting.showUpdate = this.$hasPerm('assets.change_node')
|
||||
treeSetting.showDelete = this.$hasPerm('assets.delete_node')
|
||||
},
|
||||
methods: {
|
||||
setTreeUrlQuery() {
|
||||
let str = ''
|
||||
for (const key in this.treeUrlQuery) {
|
||||
str += `${key}=${this.treeUrlQuery[key]}&`
|
||||
}
|
||||
str = str.substr(0, str.length - 1)
|
||||
|
||||
return str
|
||||
},
|
||||
decorateRMenu() {
|
||||
const show_current_asset = this.$cookie.get('show_current_asset') || '0'
|
||||
if (show_current_asset === '1') {
|
||||
$('#m_show_asset_all_children_node').css('color', '#606266')
|
||||
$('#m_show_asset_only_current_node').css('color', 'green')
|
||||
} else {
|
||||
$('#m_show_asset_all_children_node').css('color', 'green')
|
||||
$('#m_show_asset_only_current_node').css('color', '#606266')
|
||||
}
|
||||
},
|
||||
getAssetsUrl(treeNode) {
|
||||
let url = this.treeSetting?.url || this.url
|
||||
if (treeNode.meta.type === 'node') {
|
||||
const nodeId = treeNode.meta.data.id
|
||||
url = setUrlParam(url, 'node_id', nodeId)
|
||||
url = setUrlParam(url, 'asset_id', '')
|
||||
} else if (treeNode.meta.type === 'asset') {
|
||||
const assetId = treeNode.meta.data?.id || treeNode.id
|
||||
url = setUrlParam(url, 'node_id', '')
|
||||
url = setUrlParam(url, 'asset_id', assetId)
|
||||
} else if (treeNode.meta.type === 'category') {
|
||||
url = setUrlParam(url, 'category', treeNode.meta.category)
|
||||
} else if (treeNode.meta.type === 'type') {
|
||||
url = setUrlParam(url, 'category', treeNode.meta.category)
|
||||
url = setUrlParam(url, 'type', treeNode.meta._type)
|
||||
} else if (treeNode.meta.type === 'platform') {
|
||||
url = setUrlParam(url, 'platform', treeNode.id)
|
||||
}
|
||||
const query = this.setTreeUrlQuery()
|
||||
url = query ? `${url}&${query}` : url
|
||||
this.$set(this.tableConfig, 'url', url)
|
||||
setRouterQuery(this, url)
|
||||
}
|
||||
}
|
||||
}
|
||||
</script>
|
||||
|
||||
<style lang='scss' scoped>
|
||||
|
||||
</style>
|
||||
@@ -1,9 +1,13 @@
|
||||
<template>
|
||||
<DataForm
|
||||
v-if="!loading"
|
||||
:disabled="disabled"
|
||||
:fields="iFields"
|
||||
:form="value"
|
||||
style="margin-left: -26%;margin-right: -6%"
|
||||
v-bind="kwargs"
|
||||
@change="updateValue($event)"
|
||||
@input="updateValue($event)"
|
||||
v-on="$listeners"
|
||||
/>
|
||||
</template>
|
||||
@@ -28,10 +32,16 @@ export default {
|
||||
errors: {
|
||||
type: [Object, String],
|
||||
default: ''
|
||||
},
|
||||
disabled: {
|
||||
type: Boolean,
|
||||
default: false
|
||||
}
|
||||
},
|
||||
data() {
|
||||
return {
|
||||
loading: false,
|
||||
formJson: JSON.stringify(this.value),
|
||||
kwargs: {
|
||||
hasReset: false,
|
||||
hasSaveContinue: false,
|
||||
@@ -60,7 +70,26 @@ export default {
|
||||
return fields
|
||||
}
|
||||
},
|
||||
watch: {
|
||||
value: {
|
||||
handler(val) {
|
||||
const valJson = JSON.stringify(val)
|
||||
// 如果不想等,证明是 value 自己变化导致的, 需要重新渲染
|
||||
if (valJson !== this.formJson) {
|
||||
this.loading = true
|
||||
setTimeout(() => {
|
||||
this.loading = false
|
||||
}, 10)
|
||||
}
|
||||
},
|
||||
deep: true
|
||||
}
|
||||
},
|
||||
methods: {
|
||||
updateValue(val) {
|
||||
this.formJson = JSON.stringify(val)
|
||||
this.$emit('input', val)
|
||||
},
|
||||
objectToString(obj) {
|
||||
let data = ''
|
||||
// eslint-disable-next-line prefer-const
|
||||
|
||||
@@ -1,13 +1,24 @@
|
||||
<template>
|
||||
<DataForm ref="dataForm" v-loading="loading" :fields="totalFields" :form="iForm" v-bind="$attrs" v-on="$listeners">
|
||||
<FormGroupHeader
|
||||
<DataForm
|
||||
v-if="!loading"
|
||||
ref="dataForm"
|
||||
:fields="totalFields"
|
||||
:form="iForm"
|
||||
v-bind="$attrs"
|
||||
v-on="$listeners"
|
||||
>
|
||||
<span
|
||||
v-for="(group, i) in groups"
|
||||
:slot="'id:'+group.name"
|
||||
:key="'group-'+group.name"
|
||||
:group="group"
|
||||
:index="i"
|
||||
:line="i !== 0"
|
||||
/>
|
||||
:slot="'id:'+group.name"
|
||||
>
|
||||
<FormGroupHeader
|
||||
v-if="!groupHidden(group, i)"
|
||||
:group="group"
|
||||
:index="i"
|
||||
:line="i !== 0 && !groupHidden(groups[i - 1], i - 1)"
|
||||
/>
|
||||
</span>
|
||||
</DataForm>
|
||||
</template>
|
||||
|
||||
@@ -15,6 +26,7 @@
|
||||
import DataForm from '../DataForm'
|
||||
import FormGroupHeader from '@/components/FormGroupHeader'
|
||||
import { FormFieldGenerator } from '@/components/AutoDataForm/utils'
|
||||
|
||||
export default {
|
||||
name: 'AutoDataForm',
|
||||
components: {
|
||||
@@ -51,32 +63,52 @@ export default {
|
||||
totalFields: [],
|
||||
loading: true,
|
||||
groups: [],
|
||||
iForm: this.form,
|
||||
errors: {}
|
||||
}
|
||||
},
|
||||
computed: {
|
||||
iForm() {
|
||||
const iForm = {}
|
||||
Object.entries(this.form).forEach(([key, value]) => {
|
||||
// 初始值是 choice 对象
|
||||
if (value && typeof value === 'object' && value.label && value.value !== undefined) {
|
||||
iForm[key] = value.value
|
||||
} else if (Array.isArray(value) && value.length > 0 && typeof value[0] === 'object' &&
|
||||
value[0].label && value[0].value !== undefined) {
|
||||
iForm[key] = value.map(item => item.value)
|
||||
} else {
|
||||
iForm[key] = value
|
||||
}
|
||||
})
|
||||
return iForm
|
||||
}
|
||||
},
|
||||
mounted() {
|
||||
this.optionUrlMetaAndGenerateColumns()
|
||||
},
|
||||
methods: {
|
||||
optionUrlMetaAndGenerateColumns() {
|
||||
this.$store.dispatch('common/getUrlMeta', { url: this.url }).then(data => {
|
||||
this.remoteMeta = data.actions[this.method.toUpperCase()] || {}
|
||||
this.generateColumns()
|
||||
this.cleanFormValue()
|
||||
}).catch(err => {
|
||||
this.$log.error(err)
|
||||
}).finally(() => {
|
||||
this.loading = false
|
||||
})
|
||||
async optionUrlMetaAndGenerateColumns() {
|
||||
let data = { actions: {}}
|
||||
if (this.url) {
|
||||
data = await this.$store.dispatch('common/getUrlMeta', { url: this.url })
|
||||
}
|
||||
this.remoteMeta = data.actions[this.method.toUpperCase()] || {}
|
||||
this.$emit('afterRemoteMeta', this.remoteMeta)
|
||||
this.generateColumns()
|
||||
this.$emit('afterGenerateColumns', this.totalFields)
|
||||
this.cleanFormValue()
|
||||
this.loading = false
|
||||
},
|
||||
generateColumns() {
|
||||
const generator = new FormFieldGenerator()
|
||||
const generator = new FormFieldGenerator(this.$emit)
|
||||
this.totalFields = generator.generateFields(this.fields, this.fieldsMeta, this.remoteMeta)
|
||||
this.groups = generator.groups
|
||||
this.$log.debug('Total fields: ', this.totalFields)
|
||||
},
|
||||
_cleanFormValue(form, remoteMeta) {
|
||||
if (!form) {
|
||||
form = {}
|
||||
}
|
||||
for (const [k, v] of Object.entries(remoteMeta)) {
|
||||
let valueSet = form[k]
|
||||
if (v.type === 'nested object' && v.children) {
|
||||
@@ -114,6 +146,18 @@ export default {
|
||||
} else {
|
||||
field.attrs.error = error
|
||||
}
|
||||
},
|
||||
groupHidden(group, i) {
|
||||
for (const field of group.fields) {
|
||||
let hidden = field.hidden
|
||||
if (typeof hidden === 'function') {
|
||||
hidden = hidden(this.iForm)
|
||||
}
|
||||
if (!hidden) {
|
||||
return false
|
||||
}
|
||||
}
|
||||
return true
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -1,26 +1,37 @@
|
||||
import Vue from 'vue'
|
||||
import Select2 from '@/components/FormFields/Select2'
|
||||
import ObjectSelect2 from '@/components/FormFields/NestedObjectSelect2'
|
||||
import NestedField from '@/components/AutoDataForm/components/NestedField'
|
||||
import Swicher from '@/components/FormFields/Swicher'
|
||||
import Switcher from '@/components/FormFields/Switcher'
|
||||
import rules from '@/components/DataForm/rules'
|
||||
import BasicTree from '@/components/FormFields/BasicTree'
|
||||
import JsonEditor from '@/components/FormFields/JsonEditor'
|
||||
import { assignIfNot } from '@/utils/common'
|
||||
import TagInput from '@/components/FormFields/TagInput.vue'
|
||||
|
||||
export class FormFieldGenerator {
|
||||
constructor() {
|
||||
constructor(emit) {
|
||||
this.$emite = emit
|
||||
this.groups = []
|
||||
}
|
||||
|
||||
generateFieldByType(type, field, fieldMeta, fieldRemoteMeta) {
|
||||
switch (type) {
|
||||
case 'labeled_choice':
|
||||
case 'choice':
|
||||
type = 'radio-group'
|
||||
if (!fieldRemoteMeta.read_only) {
|
||||
field.options = fieldRemoteMeta.choices.map(v => {
|
||||
return { label: v.display_name, value: v.value }
|
||||
})
|
||||
// Value 处理事在 AutoDataForm 中处理的
|
||||
if (!fieldRemoteMeta['read_only']) {
|
||||
field.options = fieldRemoteMeta.choices
|
||||
}
|
||||
type = 'radio-group'
|
||||
break
|
||||
case 'multiple choice':
|
||||
field.el.choices = fieldRemoteMeta['choices']
|
||||
field.options = fieldRemoteMeta.choices
|
||||
type = 'checkbox-group'
|
||||
break
|
||||
case 'tree':
|
||||
field.el.tree = fieldRemoteMeta.tree
|
||||
field.component = BasicTree
|
||||
break
|
||||
case 'datetime':
|
||||
type = 'date-picker'
|
||||
@@ -28,12 +39,19 @@ export class FormFieldGenerator {
|
||||
type: 'datetime'
|
||||
}
|
||||
break
|
||||
case 'json':
|
||||
type = 'json-editor'
|
||||
field.component = JsonEditor
|
||||
break
|
||||
case 'field':
|
||||
type = ''
|
||||
field.component = Select2
|
||||
if (fieldRemoteMeta.required) {
|
||||
field.el.clearable = false
|
||||
}
|
||||
if (fieldRemoteMeta.child && fieldRemoteMeta.child.type === 'nested object') {
|
||||
field.component = ObjectSelect2
|
||||
}
|
||||
break
|
||||
case 'string':
|
||||
type = 'input'
|
||||
@@ -47,40 +65,54 @@ export class FormFieldGenerator {
|
||||
break
|
||||
case 'boolean':
|
||||
type = ''
|
||||
field.component = Swicher
|
||||
field.component = Switcher
|
||||
break
|
||||
case 'list':
|
||||
type = 'input'
|
||||
field.component = TagInput
|
||||
break
|
||||
case 'object_related_field':
|
||||
field.component = ObjectSelect2
|
||||
break
|
||||
case 'm2m_related_field':
|
||||
field.component = ObjectSelect2
|
||||
break
|
||||
case 'nested object':
|
||||
type = 'nestedField'
|
||||
field.component = NestedField
|
||||
field.label = ''
|
||||
field.labelWidth = 0
|
||||
field.el = { ...field.el, ...fieldMeta }
|
||||
field.el.fields = this.generateNestFields(field, fieldMeta, fieldRemoteMeta)
|
||||
field.el.errors = {}
|
||||
Vue.$log.debug('All fields in generate: ', field.el.allFields)
|
||||
field.hidden = () => {
|
||||
const hidden = fieldMeta['hiddenFields'] || (() => field.el.fields.length === 0)
|
||||
return hidden(fieldMeta, fieldRemoteMeta, field.el.fields)
|
||||
}
|
||||
break
|
||||
default:
|
||||
type = 'input'
|
||||
break
|
||||
}
|
||||
// 上面重写了 type
|
||||
if (type === 'radio-group') {
|
||||
if (!fieldRemoteMeta.read_only) {
|
||||
const options = fieldRemoteMeta.choices.map(v => {
|
||||
return { label: v.display_name, value: v.value }
|
||||
})
|
||||
if (options.length > 4) {
|
||||
type = 'select'
|
||||
field.el.filterable = true
|
||||
}
|
||||
if (field.options.length > 4) {
|
||||
type = 'select'
|
||||
field.el.filterable = true
|
||||
}
|
||||
}
|
||||
field.type = type
|
||||
return field
|
||||
}
|
||||
|
||||
generateNestFields(field, fieldMeta, fieldRemoteMeta) {
|
||||
const fields = []
|
||||
const nestedFields = fieldMeta.fields || []
|
||||
let nestedFields = fieldMeta.fields || []
|
||||
const nestedFieldsMeta = fieldMeta.fieldsMeta || {}
|
||||
const nestedFieldsRemoteMeta = fieldRemoteMeta.children || {}
|
||||
if (nestedFields === '__all__') {
|
||||
nestedFields = Object.keys(nestedFieldsRemoteMeta)
|
||||
}
|
||||
for (const name of nestedFields) {
|
||||
const f = this.generateField(name, nestedFieldsMeta, nestedFieldsRemoteMeta)
|
||||
fields.push(f)
|
||||
@@ -88,6 +120,7 @@ export class FormFieldGenerator {
|
||||
Vue.$log.debug('NestFields: ', fields)
|
||||
return fields
|
||||
}
|
||||
|
||||
generateFieldByName(name, field) {
|
||||
switch (name) {
|
||||
case 'email':
|
||||
@@ -102,6 +135,7 @@ export class FormFieldGenerator {
|
||||
}
|
||||
return field
|
||||
}
|
||||
|
||||
generateFieldByOther(field, fieldMeta, fieldRemoteMeta) {
|
||||
const filedRules = field.rules || []
|
||||
if (fieldRemoteMeta.required) {
|
||||
@@ -111,15 +145,20 @@ export class FormFieldGenerator {
|
||||
filedRules.push(rules.RequiredChange)
|
||||
}
|
||||
}
|
||||
// 一些 field 有 choices 但不是 choiceField
|
||||
if (fieldRemoteMeta.choices && field.type.indexOf('choice') === -1) {
|
||||
field.el.choices = fieldRemoteMeta.choices
|
||||
}
|
||||
field.rules = filedRules
|
||||
return field
|
||||
}
|
||||
|
||||
generateField(name, fieldsMeta, remoteFieldsMeta) {
|
||||
let field = { id: name, prop: name, el: {}, attrs: {}, rules: [] }
|
||||
const remoteFieldMeta = remoteFieldsMeta[name] || {}
|
||||
const fieldMeta = fieldsMeta[name] || {}
|
||||
field.label = remoteFieldMeta.label
|
||||
field.helpText = remoteFieldMeta.help_text
|
||||
field.helpText = remoteFieldMeta['help_text']
|
||||
field = this.generateFieldByType(remoteFieldMeta.type, field, fieldMeta, remoteFieldMeta)
|
||||
field = this.generateFieldByName(name, field)
|
||||
field = this.generateFieldByOther(field, fieldMeta, remoteFieldMeta)
|
||||
@@ -132,18 +171,25 @@ export class FormFieldGenerator {
|
||||
// Vue.$log.debug('Generate field: ', name, field)
|
||||
return field
|
||||
}
|
||||
|
||||
generateFieldGroup(field, fieldsMeta, remoteFieldsMeta) {
|
||||
const [groupTitle, fields] = field
|
||||
this.groups.push({
|
||||
const _fields = this.generateFields(fields, fieldsMeta, remoteFieldsMeta)
|
||||
const group = {
|
||||
id: groupTitle,
|
||||
title: groupTitle,
|
||||
name: fields[0],
|
||||
fields: fields
|
||||
})
|
||||
return this.generateFields(fields, fieldsMeta, remoteFieldsMeta)
|
||||
fields: _fields,
|
||||
name: _fields[0]?.id
|
||||
}
|
||||
this.groups.push(group)
|
||||
return _fields
|
||||
}
|
||||
|
||||
generateFields(_fields, fieldsMeta, remoteFieldsMeta) {
|
||||
let fields = []
|
||||
if (_fields === '__all__') {
|
||||
_fields = Object.keys(remoteFieldsMeta)
|
||||
}
|
||||
for (let field of _fields) {
|
||||
if (field instanceof Array) {
|
||||
const items = this.generateFieldGroup(field, fieldsMeta, remoteFieldsMeta)
|
||||
|
||||
@@ -33,7 +33,8 @@ export default {
|
||||
},
|
||||
computed: {
|
||||
iOption() {
|
||||
return this.options.concat(this.internalOptions)
|
||||
const options = this.options.concat(this.internalOptions)
|
||||
return _.uniqWith(options, _.isEqual)
|
||||
}
|
||||
},
|
||||
watch: {
|
||||
@@ -67,16 +68,16 @@ export default {
|
||||
type: field.type,
|
||||
value: name
|
||||
}
|
||||
if (field.type === 'choice' && field.choices) {
|
||||
if (['choice', 'labeled_choice'].indexOf(field.type) > -1 && field.choices) {
|
||||
option.children = field.choices.map(item => {
|
||||
if (typeof (item.value) === 'boolean') {
|
||||
if (item.value) {
|
||||
return { label: item.display_name, value: 'True' }
|
||||
return { label: item.label, value: 'True' }
|
||||
} else {
|
||||
return { label: item.display_name, value: 'False' }
|
||||
return { label: item.label, value: 'False' }
|
||||
}
|
||||
}
|
||||
return { label: item.display_name, value: item.value }
|
||||
return { label: item.label, value: item.value }
|
||||
})
|
||||
}
|
||||
if (field.type === 'boolean') {
|
||||
|
||||
@@ -1,12 +1,13 @@
|
||||
<template>
|
||||
<Dialog
|
||||
v-if="showColumnSettingPopover"
|
||||
:title="$t('common.CustomCol')"
|
||||
:visible.sync="showColumnSettingPopover"
|
||||
:cancel-title="$tc('common.RestoreDefault')"
|
||||
:destroy-on-close="true"
|
||||
:show-cancel="false"
|
||||
width="35%"
|
||||
:title="$tc('common.CustomCol')"
|
||||
:visible.sync="showColumnSettingPopover"
|
||||
top="10%"
|
||||
width="50%"
|
||||
@cancel="restoreDefault()"
|
||||
@confirm="handleColumnConfirm()"
|
||||
>
|
||||
<el-alert type="success">
|
||||
@@ -23,24 +24,20 @@
|
||||
style="margin-top:5px;"
|
||||
>
|
||||
<el-checkbox
|
||||
:disabled="item.prop==='actions' || minColumns.indexOf(item.prop)!==-1"
|
||||
:label="item.prop"
|
||||
:disabled="
|
||||
item.prop==='id' ||
|
||||
item.prop==='actions' ||
|
||||
minColumns.indexOf(item.prop)!==-1
|
||||
"
|
||||
>
|
||||
{{ item.label }}
|
||||
</el-checkbox>
|
||||
</el-col>
|
||||
</el-row>
|
||||
|
||||
</el-checkbox-group>
|
||||
</Dialog>
|
||||
</template>
|
||||
|
||||
<script>
|
||||
import Dialog from '@/components/Dialog/index'
|
||||
|
||||
export default {
|
||||
name: 'ColumnSettingPopover',
|
||||
components: {
|
||||
@@ -82,6 +79,10 @@ export default {
|
||||
handleColumnConfirm() {
|
||||
this.showColumnSettingPopover = false
|
||||
this.$emit('columnsUpdate', { columns: this.iCurrentColumns, url: this.url })
|
||||
},
|
||||
restoreDefault() {
|
||||
this.showColumnSettingPopover = false
|
||||
this.$emit('columnsUpdate', { columns: null, url: this.url })
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -11,8 +11,8 @@
|
||||
/>
|
||||
<ColumnSettingPopover
|
||||
:current-columns="popoverColumns.currentCols"
|
||||
:total-columns-list="popoverColumns.totalColumnsList"
|
||||
:min-columns="popoverColumns.minCols"
|
||||
:total-columns-list="popoverColumns.totalColumnsList"
|
||||
:url="config.url"
|
||||
@columnsUpdate="handlePopoverColumnsChange"
|
||||
/>
|
||||
@@ -22,15 +22,18 @@
|
||||
<script type="text/jsx">
|
||||
import DataTable from '../DataTable'
|
||||
import {
|
||||
ActionsFormatter,
|
||||
ArrayFormatter,
|
||||
ChoicesFormatter,
|
||||
DateFormatter,
|
||||
DetailFormatter,
|
||||
DisplayFormatter,
|
||||
ActionsFormatter,
|
||||
ChoicesFormatter
|
||||
ObjectRelatedFormatter
|
||||
} from '@/components/TableFormatters'
|
||||
import i18n from '@/i18n/i18n'
|
||||
import { newURL, replaceAllUUID } from '@/utils/common'
|
||||
import ColumnSettingPopover from './components/ColumnSettingPopover'
|
||||
import { newURL } from '@/utils/common'
|
||||
|
||||
export default {
|
||||
name: 'AutoDataTable',
|
||||
components: {
|
||||
@@ -63,13 +66,12 @@ export default {
|
||||
}
|
||||
}
|
||||
},
|
||||
computed: {
|
||||
},
|
||||
computed: {},
|
||||
watch: {
|
||||
config: {
|
||||
handler(iNew) {
|
||||
handler: function(iNew, iOld) {
|
||||
this.optionUrlMetaAndGenCols()
|
||||
this.$log.debug('AutoDataTable Config change found')
|
||||
this.$log.debug('AutoDataTable Config change found: ')
|
||||
},
|
||||
deep: true
|
||||
}
|
||||
@@ -79,8 +81,12 @@ export default {
|
||||
},
|
||||
methods: {
|
||||
async optionUrlMetaAndGenCols() {
|
||||
if (this.config.url === '') { return }
|
||||
const url = (this.config.url.indexOf('?') === -1) ? `${this.config.url}?draw=1&display=1` : `${this.config.url}&draw=1&display=1`
|
||||
if (this.config.url === '') {
|
||||
return
|
||||
}
|
||||
const url = (this.config.url.indexOf('?') === -1)
|
||||
? `${this.config.url}?draw=1&display=1`
|
||||
: `${this.config.url}&draw=1&display=1`
|
||||
this.$store.dispatch('common/getUrlMeta', { url: url }).then(data => {
|
||||
const method = this.method.toUpperCase()
|
||||
this.meta = data.actions && data.actions[method] ? data.actions[method] : {}
|
||||
@@ -117,7 +123,22 @@ export default {
|
||||
case 'is_valid':
|
||||
col.label = i18n.t('common.Validity')
|
||||
col.formatter = ChoicesFormatter
|
||||
col.align = 'center'
|
||||
col.formatterArgs = {
|
||||
textChoices: {
|
||||
true: i18n.t('common.Yes'),
|
||||
false: i18n.t('common.No')
|
||||
}
|
||||
}
|
||||
col.width = '80px'
|
||||
break
|
||||
case 'is_active':
|
||||
col.formatter = ChoicesFormatter
|
||||
col.formatterArgs = {
|
||||
textChoices: {
|
||||
true: i18n.t('common.Active'),
|
||||
false: i18n.t('common.Inactive')
|
||||
}
|
||||
}
|
||||
col.width = '80px'
|
||||
break
|
||||
case 'datetime':
|
||||
@@ -129,21 +150,41 @@ export default {
|
||||
}
|
||||
return col
|
||||
},
|
||||
generateColumnByType(type, col) {
|
||||
generateColumnByType(type, col, meta) {
|
||||
switch (type) {
|
||||
case 'choice':
|
||||
col.sortable = 'custom'
|
||||
col.formatter = DisplayFormatter
|
||||
break
|
||||
case 'labeled_choice':
|
||||
col.sortable = 'custom'
|
||||
col.formatter = ChoicesFormatter
|
||||
break
|
||||
case 'boolean':
|
||||
col.formatter = ChoicesFormatter
|
||||
col.align = 'center'
|
||||
col.width = '80px'
|
||||
break
|
||||
case 'datetime':
|
||||
col.formatter = DateFormatter
|
||||
col.width = '160px'
|
||||
break
|
||||
case 'object_related_field':
|
||||
col.formatter = ObjectRelatedFormatter
|
||||
break
|
||||
case 'm2m_related_field':
|
||||
col.formatter = ObjectRelatedFormatter
|
||||
break
|
||||
case 'list':
|
||||
col.formatter = ArrayFormatter
|
||||
break
|
||||
case 'json':
|
||||
case 'field':
|
||||
if (meta.child && meta.child.type === 'nested object') {
|
||||
col.formatter = ObjectRelatedFormatter
|
||||
}
|
||||
break
|
||||
}
|
||||
// this.$log.debug('Field: ', type, col.prop, col)
|
||||
return col
|
||||
},
|
||||
addHelpTipsIfNeed(col) {
|
||||
@@ -155,9 +196,9 @@ export default {
|
||||
return (
|
||||
<span>{column.label}
|
||||
<el-tooltip placement='bottom' effect='light' popperClass='help-tips'>
|
||||
<div slot='content' domPropsInnerHTML={helpTips} />
|
||||
<div slot='content' domPropsInnerHTML={helpTips}/>
|
||||
<el-button style='padding: 0'>
|
||||
<i class='fa fa-info-circle' />
|
||||
<i class='fa fa-info-circle'/>
|
||||
</el-button>
|
||||
</el-tooltip>
|
||||
</span>
|
||||
@@ -183,12 +224,12 @@ export default {
|
||||
col.filters = column.choices.map(item => {
|
||||
if (typeof (item.value) === 'boolean') {
|
||||
if (item.value) {
|
||||
return { text: item['display_name'], value: 'True' }
|
||||
return { text: item['label'], value: 'True' }
|
||||
} else {
|
||||
return { text: item['display_name'], value: 'False' }
|
||||
return { text: item['label'], value: 'False' }
|
||||
}
|
||||
}
|
||||
return { text: item['display_name'], value: item.value }
|
||||
return { text: item['label'], value: item.value }
|
||||
})
|
||||
col.sortable = false
|
||||
col['column-key'] = col.prop
|
||||
@@ -196,22 +237,65 @@ export default {
|
||||
}
|
||||
return col
|
||||
},
|
||||
addOrderingIfNeed(col) {
|
||||
if (col.prop) {
|
||||
const column = this.meta[col.prop] || {}
|
||||
if (column.order) {
|
||||
col.sortable = 'custom'
|
||||
col['column-key'] = col.prop
|
||||
}
|
||||
}
|
||||
return col
|
||||
},
|
||||
setDefaultFormatterIfNeed(col) {
|
||||
if (!col.formatter) {
|
||||
col.formatter = (row, column, cellValue) => {
|
||||
let value = cellValue
|
||||
let padding = '0'
|
||||
const excludes = [undefined, null, '']
|
||||
if (excludes.indexOf(value) !== -1) {
|
||||
padding = '6px'
|
||||
value = '-'
|
||||
}
|
||||
return <span style={{ marginLeft: padding }}>{value}</span>
|
||||
}
|
||||
}
|
||||
return col
|
||||
},
|
||||
|
||||
generateColumn(name) {
|
||||
const colMeta = this.meta[name] || {}
|
||||
const customMeta = this.config.columnsMeta ? this.config.columnsMeta[name] : {}
|
||||
let col = { prop: name, label: colMeta.label }
|
||||
let col = { prop: name, label: colMeta.label, showOverflowTooltip: true }
|
||||
|
||||
col = this.generateColumnByName(name, col)
|
||||
col = this.generateColumnByType(colMeta.type, col)
|
||||
col = this.generateColumnByType(colMeta.type, col, colMeta)
|
||||
col = this.setDefaultFormatterIfNeed(col)
|
||||
col = Object.assign(col, customMeta)
|
||||
col = this.addHelpTipsIfNeed(col)
|
||||
col = this.addFilterIfNeed(col)
|
||||
col = this.addOrderingIfNeed(col)
|
||||
return col
|
||||
},
|
||||
generateTotalColumns() {
|
||||
const config = _.cloneDeep(this.config)
|
||||
let columns = []
|
||||
for (let col of config.columns) {
|
||||
const allColumnNames = Object.entries(this.meta)
|
||||
.filter(([name, meta]) => !meta['write_only'])
|
||||
.map(([name, meta]) => name)
|
||||
.concat(config.columnsExtra || [])
|
||||
|
||||
let configColumns = config.columns || allColumnNames
|
||||
const columnsExclude = config.columnsExclude || []
|
||||
configColumns = configColumns.filter(item => !columnsExclude.includes(item))
|
||||
|
||||
// 解决后端 API 返回字段中包含 actions 的问题;
|
||||
const hasColumnActions = configColumns.findIndex(item => item?.prop === 'actions') !== -1
|
||||
if (!hasColumnActions) {
|
||||
configColumns = [...configColumns.filter(i => i !== 'actions'), 'actions']
|
||||
}
|
||||
|
||||
for (let col of configColumns) {
|
||||
if (typeof col === 'object') {
|
||||
columns.push(col)
|
||||
} else if (typeof col === 'string') {
|
||||
@@ -219,7 +303,11 @@ export default {
|
||||
columns.push(col)
|
||||
}
|
||||
}
|
||||
|
||||
columns = columns.filter(item => {
|
||||
if (item?.showFullContent) {
|
||||
item.className = 'show-full-content'
|
||||
}
|
||||
let has = item.has
|
||||
if (has === undefined) {
|
||||
has = true
|
||||
@@ -246,13 +334,14 @@ export default {
|
||||
|
||||
// 最小列
|
||||
const minColumnsNames = _.get(this.iConfig, 'columnsShow.min', ['actions', 'id'])
|
||||
.filter(n => defaultColumnsNames.indexOf(n) > -1)
|
||||
.filter(n => totalColumnsNames.includes(n))
|
||||
|
||||
// 应该显示的列
|
||||
const _tableConfig = localStorage.getItem('tableConfig')
|
||||
? JSON.parse(localStorage.getItem('tableConfig'))
|
||||
: {}
|
||||
const tableName = this.config.name || this.$route.name + '_' + newURL(this.iConfig.url).pathname
|
||||
let tableName = this.config.name || this.$route.name + '_' + newURL(this.iConfig.url).pathname
|
||||
tableName = replaceAllUUID(tableName)
|
||||
const configShowColumnsNames = _.get(_tableConfig[tableName], 'showColumns', null)
|
||||
let showColumnsNames = configShowColumnsNames || defaultColumnsNames
|
||||
if (showColumnsNames.length === 0) {
|
||||
@@ -293,11 +382,17 @@ export default {
|
||||
},
|
||||
handlePopoverColumnsChange({ columns, url }) {
|
||||
this.$log.debug('Columns change: ', columns)
|
||||
if (columns === null) {
|
||||
columns = this.cleanedColumnsShow.default
|
||||
}
|
||||
this.popoverColumns.currentCols = columns
|
||||
const _tableConfig = localStorage.getItem('tableConfig')
|
||||
? JSON.parse(localStorage.getItem('tableConfig'))
|
||||
: {}
|
||||
const tableName = this.config.name || this.$route.name + '_' + newURL(url).pathname
|
||||
let tableName = this.config.name || this.$route.name + '_' + newURL(url).pathname
|
||||
// 替换url中的uuid,避免同一个类型接口生成多个key,localStorage中的数据无法共用
|
||||
tableName = replaceAllUUID(tableName)
|
||||
|
||||
_tableConfig[tableName] = {
|
||||
'showColumns': columns
|
||||
}
|
||||
|
||||
@@ -2,13 +2,13 @@
|
||||
<DataZTree ref="dataztree" :setting="treeSetting" class="data-z-tree" v-on="$listeners">
|
||||
<slot v-if="treeSetting.hasRightMenu" slot="rMenu">
|
||||
<li v-if="treeSetting.showCreate" id="m_create" class="rmenu" tabindex="-1" @click="createTreeNode">
|
||||
<i class="fa fa-plus-square-o" /> {{ this.$t('tree.CreateNode') }}
|
||||
<i class="fa fa-plus-square-o" /> {{ this.$t('tree.CreateNode') }}
|
||||
</li>
|
||||
<li v-if="treeSetting.showUpdate" id="m_edit" class="rmenu" tabindex="-1" @click="editTreeNode">
|
||||
<i class="fa fa-pencil-square-o" /> {{ this.$t('tree.RenameNode') }}
|
||||
<i class="fa fa-pencil-square-o" /> {{ this.$t('tree.RenameNode') }}
|
||||
</li>
|
||||
<li v-if="treeSetting.showDelete" id="m_del" class="rmenu" tabindex="-1" @click="removeTreeNode">
|
||||
<i class="fa fa-minus-square" /> {{ this.$t('tree.DeleteNode') }}
|
||||
<i class="fa fa-minus-square" /> {{ this.$t('tree.DeleteNode') }}
|
||||
</li>
|
||||
<slot name="rMenu" />
|
||||
</slot>
|
||||
@@ -39,12 +39,12 @@ export default {
|
||||
showDelete: true,
|
||||
showUpdate: true,
|
||||
showSearch: false,
|
||||
// 自定义header
|
||||
customTreeHeader: false,
|
||||
customTreeHeaderName: this.$t('assets.AssetTree'),
|
||||
async: {
|
||||
enable: true,
|
||||
url: (process.env.VUE_APP_ENV === 'production') ? (`${this.setting.treeUrl}`) : (`${process.env.VUE_APP_BASE_API}${this.setting.treeUrl}`),
|
||||
url: (process.env.VUE_APP_ENV === 'production')
|
||||
? (`${this.setting.treeUrl}`)
|
||||
: (`${process.env.VUE_APP_BASE_API}${this.setting.treeUrl}`),
|
||||
autoParam: ['id=key', 'name=n', 'level=lv'],
|
||||
type: 'get',
|
||||
headers: {
|
||||
@@ -128,9 +128,9 @@ export default {
|
||||
query['asset'] = ''
|
||||
url = `${this.setting.url}${combinator}node_id=${objectId}&show_current_asset=${show_current_asset}`
|
||||
} else if (treeNode.meta.type === 'asset') {
|
||||
query['asset'] = treeNode.meta.data.id
|
||||
query['asset'] = treeNode.meta.data?.id || treeNode.id
|
||||
query['node'] = ''
|
||||
url = `${this.setting.url}${combinator}asset_id=${objectId}&show_current_asset=${show_current_asset}`
|
||||
url = `${this.setting.url}${combinator}asset_id=${query.asset}&show_current_asset=${show_current_asset}`
|
||||
}
|
||||
this.$router.push({ query })
|
||||
this.$emit('urlChange', url)
|
||||
@@ -144,22 +144,20 @@ export default {
|
||||
this.$axios.delete(
|
||||
`${this.treeSetting.nodeUrl}${currentNode.meta.data.id}/`
|
||||
).then(() => {
|
||||
this.$message.success(this.$t('common.deleteSuccessMsg'))
|
||||
this.$message.success(this.$tc('common.deleteSuccessMsg'))
|
||||
this.zTree.removeNode(currentNode)
|
||||
this.refreshTree()
|
||||
}).catch(() => {
|
||||
// this.$message.error(this.$t('common.deleteErrorMsg') + ' ' + error)
|
||||
// this.$message.error(this.$tc('common.deleteErrorMsg') + ' ' + error)
|
||||
})
|
||||
},
|
||||
onRename: function(event, treeId, treeNode, isCancel) {
|
||||
const url = `${this.treeSetting.nodeUrl}${this.currentNodeId}/`
|
||||
const currentNodeId = this.currentNodeId || treeNode.meta.data?.id || ''
|
||||
const url = `${this.treeSetting.nodeUrl}${currentNodeId}/`
|
||||
if (isCancel) {
|
||||
return
|
||||
}
|
||||
this.$axios.patch(
|
||||
url,
|
||||
{ 'value': treeNode.name }
|
||||
).then(res => {
|
||||
this.$axios.patch(url, { 'value': treeNode.name }).then(res => {
|
||||
let assetsAmount = treeNode.meta.data['assetsAmount']
|
||||
if (!assetsAmount) {
|
||||
assetsAmount = 0
|
||||
@@ -167,8 +165,10 @@ export default {
|
||||
treeNode.name = treeNode.name + ' (' + assetsAmount + ')'
|
||||
treeNode.meta.data = res
|
||||
this.zTree.updateNode(treeNode)
|
||||
this.$message.success(this.$t('common.updateSuccessMsg'))
|
||||
}).finally(() => { this.refreshTree() })
|
||||
this.$message.success(this.$tc('common.updateSuccessMsg'))
|
||||
}).finally(() => {
|
||||
this.refreshTree()
|
||||
})
|
||||
},
|
||||
onBodyMouseDown: function(event) {
|
||||
const rMenuID = this.$refs.dataztree.$refs.ztree.iRMenuID
|
||||
@@ -208,6 +208,9 @@ export default {
|
||||
this.showRMenu('root', event.clientX, event.clientY)
|
||||
} else if (treeNode && !treeNode.noR) {
|
||||
this.zTree.selectNode(treeNode)
|
||||
if (treeNode.meta?.data?.id) {
|
||||
this.currentNodeId = treeNode.meta.data.id
|
||||
}
|
||||
this.showRMenu('node', event.clientX, event.clientY)
|
||||
}
|
||||
},
|
||||
@@ -234,9 +237,9 @@ export default {
|
||||
nodes: treeNodesIds
|
||||
}
|
||||
).then((res) => {
|
||||
this.$message.success(this.$t('common.updateSuccessMsg'))
|
||||
this.$message.success(this.$tc('common.updateSuccessMsg'))
|
||||
}).catch(error => {
|
||||
this.$message.error(this.$t('common.updateErrorMsg' + ' ' + error))
|
||||
this.$message.error(this.$tc('common.updateErrorMsg' + ' ' + error))
|
||||
}).finally()
|
||||
},
|
||||
createTreeNode: function() {
|
||||
@@ -264,9 +267,9 @@ export default {
|
||||
const node = this.zTree.getNodeByParam('id', newNode.id, parentNode)
|
||||
this.currentNodeId = node.meta.data.id || newNode.id
|
||||
this.zTree.editName(node)
|
||||
this.$message.success(this.$t('common.createSuccessMsg'))
|
||||
this.$message.success(this.$tc('common.createSuccessMsg'))
|
||||
}).catch(error => {
|
||||
this.$message.error(this.$t('common.createErrorMsg') + ' ' + error)
|
||||
this.$message.error(this.$tc('common.createErrorMsg') + ' ' + error)
|
||||
})
|
||||
},
|
||||
refresh: function() {
|
||||
@@ -285,30 +288,32 @@ export default {
|
||||
</script>
|
||||
|
||||
<style scoped>
|
||||
.rmenu {
|
||||
font-size: 12px;
|
||||
padding: 0 16px;
|
||||
position: relative;
|
||||
white-space: nowrap;
|
||||
overflow: hidden;
|
||||
text-overflow: ellipsis;
|
||||
color: #606266;
|
||||
height: 24px;
|
||||
line-height: 24px;
|
||||
box-sizing: border-box;
|
||||
cursor: pointer;
|
||||
}
|
||||
.rmenu {
|
||||
font-size: 12px;
|
||||
padding: 0 16px;
|
||||
position: relative;
|
||||
white-space: nowrap;
|
||||
overflow: hidden;
|
||||
text-overflow: ellipsis;
|
||||
color: #606266;
|
||||
height: 24px;
|
||||
line-height: 24px;
|
||||
box-sizing: border-box;
|
||||
cursor: pointer;
|
||||
}
|
||||
|
||||
.rmenu > a:hover, .dropdown-menu > a:focus {
|
||||
color: #262626;
|
||||
text-decoration: none;
|
||||
background-color: #f5f5f5;
|
||||
}
|
||||
.rmenu:hover{
|
||||
background-color: #f5f7fa;
|
||||
}
|
||||
.rmenu > a:hover, .dropdown-menu > a:focus {
|
||||
color: #262626;
|
||||
text-decoration: none;
|
||||
background-color: #f5f5f5;
|
||||
}
|
||||
|
||||
.data-z-tree >>> .fa {
|
||||
width: 10px;
|
||||
}
|
||||
.rmenu:hover {
|
||||
background-color: #f5f7fa;
|
||||
}
|
||||
|
||||
.data-z-tree >>> .fa {
|
||||
width: 10px;
|
||||
margin-right: 3px;
|
||||
}
|
||||
</style>
|
||||
|
||||
188
src/components/AutomationParams/index.vue
Normal file
188
src/components/AutomationParams/index.vue
Normal file
@@ -0,0 +1,188 @@
|
||||
<template>
|
||||
<div>
|
||||
<div>
|
||||
<el-button
|
||||
size="mini"
|
||||
type="primary"
|
||||
:disabled="isDisabled"
|
||||
@click="onOpenDialog"
|
||||
>{{ $tc('common.Setting') }}</el-button>
|
||||
</div>
|
||||
<Dialog
|
||||
v-if="visible"
|
||||
width="60%"
|
||||
:visible.sync="visible"
|
||||
:title="title"
|
||||
:show-cancel="false"
|
||||
:show-confirm="false"
|
||||
:destroy-on-close="true"
|
||||
v-bind="$attrs"
|
||||
v-on="$listeners"
|
||||
>
|
||||
<AutoDataForm
|
||||
ref="autoDataForm"
|
||||
:form="form"
|
||||
class="data-form"
|
||||
v-bind="config"
|
||||
@submit="onSubmit"
|
||||
/>
|
||||
</Dialog>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<script>
|
||||
import { Dialog, AutoDataForm } from '@/components'
|
||||
|
||||
export default {
|
||||
componentName: 'AutomationParams',
|
||||
components: {
|
||||
Dialog,
|
||||
AutoDataForm
|
||||
},
|
||||
props: {
|
||||
value: {
|
||||
type: Object,
|
||||
default: () => ({})
|
||||
},
|
||||
title: {
|
||||
type: String,
|
||||
default: function() {
|
||||
return this.$t('assets.PushParams')
|
||||
}
|
||||
},
|
||||
assets: {
|
||||
type: Array,
|
||||
default: () => []
|
||||
},
|
||||
nodes: {
|
||||
type: Array,
|
||||
default: () => []
|
||||
},
|
||||
method: {
|
||||
type: String,
|
||||
default: ''
|
||||
},
|
||||
url: {
|
||||
type: String,
|
||||
default: `/api/v1/assets/platform-automation-methods/`
|
||||
}
|
||||
},
|
||||
data() {
|
||||
return {
|
||||
remoteMeta: {},
|
||||
visible: false,
|
||||
isDisabled: true,
|
||||
form: this.value,
|
||||
node_ids: this.nodes,
|
||||
asset_ids: this.assets,
|
||||
config: {
|
||||
url: this.url,
|
||||
hasSaveContinue: false,
|
||||
hasButtons: true,
|
||||
method: 'get',
|
||||
fields: [],
|
||||
fieldsMeta: {}
|
||||
}
|
||||
}
|
||||
},
|
||||
computed: {
|
||||
refForm() {
|
||||
return this.$refs.autoDataForm
|
||||
}
|
||||
},
|
||||
watch: {
|
||||
nodes: {
|
||||
handler(val) {
|
||||
this.node_ids = val
|
||||
this.onFieldChangeHandle()
|
||||
},
|
||||
deep: true
|
||||
},
|
||||
assets: {
|
||||
handler(val) {
|
||||
this.asset_ids = val
|
||||
this.onFieldChangeHandle()
|
||||
},
|
||||
deep: true
|
||||
}
|
||||
},
|
||||
created() {
|
||||
this.getUrlMeta()
|
||||
},
|
||||
methods: {
|
||||
async getUrlMeta() {
|
||||
const data = await this.$store.dispatch('common/getUrlMeta', { url: this.url })
|
||||
this.remoteMeta = data.actions[this.config.method.toUpperCase()] || {}
|
||||
},
|
||||
async getFilterPlatforms() {
|
||||
return await this.$axios.post(
|
||||
'/api/v1/assets/platforms/filter-nodes-assets/',
|
||||
{
|
||||
'node_ids': this.node_ids,
|
||||
'asset_ids': this.asset_ids
|
||||
}
|
||||
)
|
||||
},
|
||||
async onFieldChangeHandle() {
|
||||
const platforms = await this.getFilterPlatforms()
|
||||
let pushAccountMethods = platforms.map(i => i.automation[this.method])
|
||||
pushAccountMethods = _.uniq(pushAccountMethods)
|
||||
// 检测是否有可设置的推送方式
|
||||
const hasCanSettingPushMethods = _.intersection(pushAccountMethods, Object.keys(this.remoteMeta))
|
||||
this.setFormConfig(hasCanSettingPushMethods)
|
||||
if (hasCanSettingPushMethods.length > 0) {
|
||||
this.isDisabled = false
|
||||
this.$emit('input', this.form)
|
||||
} else {
|
||||
this.isDisabled = true
|
||||
this.$emit('input', {})
|
||||
}
|
||||
},
|
||||
setFormConfig(methods) {
|
||||
const newForm = {}
|
||||
const fields = []
|
||||
const fieldsMeta = {}
|
||||
this.config.fields = []
|
||||
|
||||
for (const method of methods) {
|
||||
const filterField = this.remoteMeta[method] || {}
|
||||
// 修改资产、节点时不点击设置按钮也需要获取form表单值暴露出去
|
||||
if (this.form.hasOwnProperty(method)) {
|
||||
newForm[method] = this.form[method]
|
||||
}
|
||||
fields.push([filterField.label, [method]])
|
||||
fieldsMeta[method] = {
|
||||
fields: [],
|
||||
fieldsMeta: {}
|
||||
}
|
||||
if (Object.keys(filterField?.children || {}).length > 0) {
|
||||
for (const [k, v] of Object.entries(filterField.children)) {
|
||||
const item = {
|
||||
...v,
|
||||
type: 'input'
|
||||
}
|
||||
delete item.default
|
||||
fieldsMeta[method].fields.push(k)
|
||||
fieldsMeta[method].fieldsMeta[k] = item
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
this.form = newForm
|
||||
this.config.fields = fields
|
||||
this.config.fieldsMeta = fieldsMeta
|
||||
},
|
||||
onOpenDialog() {
|
||||
this.visible = true
|
||||
},
|
||||
onSubmit(form) {
|
||||
this.visible = false
|
||||
this.$emit('input', form)
|
||||
this.$log.debug('Auto push form:', form)
|
||||
}
|
||||
}
|
||||
}
|
||||
</script>
|
||||
|
||||
<style scoped>
|
||||
</style>
|
||||
@@ -2,7 +2,7 @@
|
||||
<template>
|
||||
<div>
|
||||
<el-tabs type="border-card">
|
||||
<el-tab-pane v-if="shouldHide('min')" :label="this.$t('common.CronTab.min')">
|
||||
<el-tab-pane v-if="shouldHide('min')" :label="$tc('common.CronTab.min')">
|
||||
<CrontabMin
|
||||
ref="cronmin"
|
||||
:check="checkNumber"
|
||||
@@ -11,7 +11,7 @@
|
||||
/>
|
||||
</el-tab-pane>
|
||||
|
||||
<el-tab-pane v-if="shouldHide('hour')" :label="this.$t('common.CronTab.hour')">
|
||||
<el-tab-pane v-if="shouldHide('hour')" :label="$tc('common.CronTab.hour')">
|
||||
<CrontabHour
|
||||
ref="cronhour"
|
||||
:check="checkNumber"
|
||||
@@ -20,7 +20,7 @@
|
||||
/>
|
||||
</el-tab-pane>
|
||||
|
||||
<el-tab-pane v-if="shouldHide('day')" :label="this.$t('common.CronTab.day')">
|
||||
<el-tab-pane v-if="shouldHide('day')" :label="$tc('common.CronTab.day')">
|
||||
<CrontabDay
|
||||
ref="cronday"
|
||||
:check="checkNumber"
|
||||
@@ -29,7 +29,7 @@
|
||||
/>
|
||||
</el-tab-pane>
|
||||
|
||||
<el-tab-pane v-if="shouldHide('month')" :label="this.$t('common.CronTab.month')">
|
||||
<el-tab-pane v-if="shouldHide('month')" :label="$tc('common.CronTab.month')">
|
||||
<CrontabMonth
|
||||
ref="cronmonth"
|
||||
:check="checkNumber"
|
||||
@@ -38,7 +38,7 @@
|
||||
/>
|
||||
</el-tab-pane>
|
||||
|
||||
<el-tab-pane v-if="shouldHide('week')" :label="this.$t('common.CronTab.week')">
|
||||
<el-tab-pane v-if="shouldHide('week')" :label="$tc('common.CronTab.week')">
|
||||
<CrontabWeek
|
||||
ref="cronweek"
|
||||
:check="checkNumber"
|
||||
@@ -390,6 +390,7 @@ export default {
|
||||
text-align: center;
|
||||
margin-top: 20px;
|
||||
}
|
||||
|
||||
.popup-main {
|
||||
position: relative;
|
||||
margin: 10px auto 0;
|
||||
@@ -398,12 +399,14 @@ export default {
|
||||
font-size: 12px;
|
||||
overflow: hidden;
|
||||
}
|
||||
|
||||
.popup-title {
|
||||
overflow: hidden;
|
||||
line-height: 34px;
|
||||
padding-top: 6px;
|
||||
background: #f2f2f2;
|
||||
}
|
||||
|
||||
.popup-result {
|
||||
position: relative;
|
||||
box-sizing: border-box;
|
||||
@@ -413,6 +416,7 @@ export default {
|
||||
border: 1px solid #dcdfe6;
|
||||
box-shadow: 0 2px 4px 0 rgb(0 0 0 / 12%), 0 0 6px 0 rgb(0 0 0 / 4%);
|
||||
}
|
||||
|
||||
.popup-result .title {
|
||||
position: absolute;
|
||||
top: -17px;
|
||||
@@ -424,11 +428,13 @@ export default {
|
||||
line-height: 30px;
|
||||
background: #fff;
|
||||
}
|
||||
|
||||
.popup-result table {
|
||||
text-align: center;
|
||||
width: 100%;
|
||||
margin: 0 auto;
|
||||
}
|
||||
|
||||
.popup-result table span {
|
||||
display: block;
|
||||
width: 100%;
|
||||
@@ -439,12 +445,14 @@ export default {
|
||||
overflow: hidden;
|
||||
border: 1px solid #e8e8e8;
|
||||
}
|
||||
|
||||
.popup-result-scroll {
|
||||
font-size: 12px;
|
||||
line-height: 24px;
|
||||
height: 10em;
|
||||
overflow-y: auto;
|
||||
}
|
||||
|
||||
.el-form-item--mini.el-form-item,
|
||||
.el-form-item--small.el-form-item {
|
||||
margin-bottom: 10px;
|
||||
|
||||
@@ -25,7 +25,13 @@
|
||||
<el-form-item>
|
||||
<el-radio v-model="radioValue" :label="7">
|
||||
{{ this.$t('common.CronTab.appoint') }}
|
||||
<el-select v-model="checkboxList" clearable :placeholder="this.$t('common.CronTab.manyChoose')" multiple style="width:100%">
|
||||
<el-select
|
||||
v-model="checkboxList"
|
||||
clearable
|
||||
:placeholder="$tc('common.CronTab.manyChoose')"
|
||||
multiple
|
||||
style="width:100%"
|
||||
>
|
||||
<el-option v-for="item in 31" :key="item" :value="item">{{ item }}</el-option>
|
||||
</el-select>
|
||||
</el-radio>
|
||||
@@ -45,7 +51,8 @@ export default {
|
||||
},
|
||||
check: {
|
||||
type: Function,
|
||||
default: () => {}
|
||||
default: () => {
|
||||
}
|
||||
}
|
||||
},
|
||||
data() {
|
||||
@@ -184,6 +191,6 @@ export default {
|
||||
|
||||
<style scoped>
|
||||
.el-form-item--small.el-form-item {
|
||||
margin-bottom: 10px;
|
||||
}
|
||||
margin-bottom: 10px;
|
||||
}
|
||||
</style>
|
||||
|
||||
@@ -25,7 +25,13 @@
|
||||
<el-form-item>
|
||||
<el-radio v-model="radioValue" :label="4">
|
||||
{{ this.$t('common.CronTab.appoint') }}
|
||||
<el-select v-model="checkboxList" clearable :placeholder="this.$t('common.CronTab.manyChoose')" multiple style="width:100%">
|
||||
<el-select
|
||||
v-model="checkboxList"
|
||||
clearable
|
||||
:placeholder="$tc('common.CronTab.manyChoose')"
|
||||
multiple
|
||||
style="width:100%"
|
||||
>
|
||||
<el-option v-for="item in 24" :key="item" :value="item-1">{{ item-1 }}</el-option>
|
||||
</el-select>
|
||||
</el-radio>
|
||||
@@ -45,7 +51,8 @@ export default {
|
||||
},
|
||||
check: {
|
||||
type: Function,
|
||||
default: () => {}
|
||||
default: () => {
|
||||
}
|
||||
}
|
||||
},
|
||||
data() {
|
||||
@@ -153,6 +160,6 @@ export default {
|
||||
|
||||
<style scoped>
|
||||
.el-form-item--small.el-form-item {
|
||||
margin-bottom: 10px
|
||||
}
|
||||
margin-bottom: 10px
|
||||
}
|
||||
</style>
|
||||
|
||||
@@ -25,7 +25,14 @@
|
||||
<el-form-item>
|
||||
<el-radio v-model="radioValue" :label="4">
|
||||
{{ this.$t('common.CronTab.appoint') }}
|
||||
<el-select v-model="checkboxList" clearable :placeholder="this.$t('common.CronTab.manyChoose')" multiple style="width:100%" size="small">
|
||||
<el-select
|
||||
v-model="checkboxList"
|
||||
clearable
|
||||
:placeholder="$tc('common.CronTab.manyChoose')"
|
||||
multiple
|
||||
style="width:100%"
|
||||
size="small"
|
||||
>
|
||||
<el-option v-for="item in 60" :key="item" :value="item-1">{{ item-1 }}</el-option>
|
||||
</el-select>
|
||||
</el-radio>
|
||||
@@ -46,7 +53,8 @@ export default {
|
||||
},
|
||||
check: {
|
||||
type: Function,
|
||||
default: () => {}
|
||||
default: () => {
|
||||
}
|
||||
}
|
||||
},
|
||||
data() {
|
||||
@@ -151,6 +159,6 @@ export default {
|
||||
|
||||
<style scoped>
|
||||
.el-form-item--small.el-form-item {
|
||||
margin-bottom: 10px;
|
||||
}
|
||||
margin-bottom: 10px;
|
||||
}
|
||||
</style>
|
||||
|
||||
@@ -25,7 +25,13 @@
|
||||
<el-form-item>
|
||||
<el-radio v-model="radioValue" :label="4">
|
||||
{{ this.$t('common.CronTab.appoint') }}
|
||||
<el-select v-model="checkboxList" clearable :placeholder="this.$t('common.CronTab.manyChoose')" multiple style="width:100%">
|
||||
<el-select
|
||||
v-model="checkboxList"
|
||||
clearable
|
||||
:placeholder="$tc('common.CronTab.manyChoose')"
|
||||
multiple
|
||||
style="width:100%"
|
||||
>
|
||||
<el-option v-for="item in 12" :key="item" :value="item">{{ item }}</el-option>
|
||||
</el-select>
|
||||
</el-radio>
|
||||
@@ -45,7 +51,8 @@ export default {
|
||||
},
|
||||
check: {
|
||||
type: Function,
|
||||
default: () => {}
|
||||
default: () => {
|
||||
}
|
||||
}
|
||||
},
|
||||
data() {
|
||||
@@ -158,6 +165,6 @@ export default {
|
||||
|
||||
<style scoped>
|
||||
.el-form-item--small.el-form-item {
|
||||
margin-bottom: 10px;
|
||||
}
|
||||
margin-bottom: 10px;
|
||||
}
|
||||
</style>
|
||||
|
||||
@@ -13,7 +13,7 @@
|
||||
|
||||
<script>
|
||||
import parser from 'cron-parser'
|
||||
import moment from 'moment'
|
||||
import { toSafeLocalDateStr } from '@/utils/common'
|
||||
export default {
|
||||
name: 'CrontabResult',
|
||||
props: {
|
||||
@@ -46,10 +46,10 @@ export default {
|
||||
const rule = 0 + ' ' + this.$options.propsData.ex
|
||||
try {
|
||||
this.resultList = []
|
||||
var interval = parser.parseExpression(rule)
|
||||
const interval = parser.parseExpression(rule)
|
||||
for (let index = 0; index < 5; index++) {
|
||||
const cur = interval.next().toString()
|
||||
this.resultList.push(moment(cur).format('YYYY-MM-DD HH:mm:ss'))
|
||||
this.resultList.push(toSafeLocalDateStr(cur))
|
||||
}
|
||||
} catch (error) {
|
||||
this.isShow = false
|
||||
|
||||
@@ -18,7 +18,13 @@
|
||||
<el-form-item>
|
||||
<el-radio v-model="radioValue" :label="6">
|
||||
{{ this.$t('common.CronTab.appoint') }}
|
||||
<el-select v-model="checkboxList" clearable :placeholder="this.$t('common.CronTab.manyChoose')" multiple style="width:100%">
|
||||
<el-select
|
||||
v-model="checkboxList"
|
||||
clearable
|
||||
:placeholder="$tc('common.CronTab.manyChoose')"
|
||||
multiple
|
||||
style="width:100%"
|
||||
>
|
||||
<el-option v-for="(item,index) of weekList" :key="index" :value="index+1">{{ item }}</el-option>
|
||||
</el-select>
|
||||
</el-radio>
|
||||
@@ -39,7 +45,8 @@ export default {
|
||||
},
|
||||
check: {
|
||||
type: Function,
|
||||
default: () => {}
|
||||
default: () => {
|
||||
}
|
||||
}
|
||||
},
|
||||
data() {
|
||||
@@ -173,6 +180,6 @@ export default {
|
||||
|
||||
<style scoped>
|
||||
.el-form-item--small.el-form-item {
|
||||
margin-bottom: 10px;
|
||||
}
|
||||
margin-bottom: 10px;
|
||||
}
|
||||
</style>
|
||||
|
||||
@@ -3,7 +3,7 @@
|
||||
<div class="box">
|
||||
<el-input v-model="input" clearable @focus="showDialog" @clear="onClear" />
|
||||
</div>
|
||||
<el-dialog :title="this.$t('common.CronTab.newCron')" :visible.sync="showCron" top="8vh" width="580px" append-to-body>
|
||||
<el-dialog :title="$tc('common.CronTab.newCron')" :visible.sync="showCron" top="8vh" width="580px" append-to-body>
|
||||
<Crontab
|
||||
:expression="expression"
|
||||
@hide="showCron = false"
|
||||
@@ -15,6 +15,7 @@
|
||||
|
||||
<script>
|
||||
import Crontab from './Crontab.vue'
|
||||
|
||||
export default {
|
||||
components: { Crontab },
|
||||
props: {
|
||||
@@ -51,5 +52,5 @@ export default {
|
||||
<style scoped>
|
||||
.el-dialog__body {
|
||||
padding: 12px 16px;
|
||||
}
|
||||
}
|
||||
</style>
|
||||
|
||||
@@ -10,7 +10,7 @@
|
||||
placement="bottom-start"
|
||||
@command="handleDropdownCallback"
|
||||
>
|
||||
<el-button :size="size" v-bind="cleanButtonAction(action)">
|
||||
<el-button class="more-action" :size="size" v-bind="cleanButtonAction(action)">
|
||||
{{ action.title }}<i class="el-icon-arrow-down el-icon--right" />
|
||||
</el-button>
|
||||
<el-dropdown-menu slot="dropdown" style="overflow: auto;max-height: 60vh">
|
||||
@@ -28,6 +28,10 @@
|
||||
:command="[option, action]"
|
||||
v-bind="option"
|
||||
>
|
||||
<span v-if="option.fa">
|
||||
<i v-if="option.fa.startsWith('fa-')" :class="'fa ' + option.fa" />
|
||||
<svg-icon v-else :icon-class="option.fa" style="font-size: 14px; margin-right: 2px; margin-left: -2px;" />
|
||||
</span>
|
||||
{{ option.title }}
|
||||
</el-dropdown-item>
|
||||
</template>
|
||||
@@ -44,7 +48,11 @@
|
||||
>
|
||||
<el-tooltip :disabled="!action.tip" :content="action.tip" placement="top">
|
||||
<span>
|
||||
<i v-if="action.fa" :class="'fa ' + action.fa" />{{ action.title }}
|
||||
<span v-if="action.fa" style="vertical-align: initial;">
|
||||
<i v-if="action.fa.startsWith('fa-')" :class="'fa ' + action.fa" />
|
||||
<svg-icon v-else :icon-class="action.fa" style="font-size: 14px;" />
|
||||
</span>
|
||||
{{ action.title }}
|
||||
</span>
|
||||
</el-tooltip>
|
||||
</el-button>
|
||||
@@ -153,7 +161,7 @@ export default {
|
||||
}
|
||||
</script>
|
||||
|
||||
<style scoped>
|
||||
<style lang="scss" scoped>
|
||||
.layout {
|
||||
display: flex;
|
||||
justify-content: center;
|
||||
@@ -181,4 +189,22 @@ export default {
|
||||
.el-button-ungroup .action-item:first-child {
|
||||
margin-left: 0;
|
||||
}
|
||||
::v-deep .more-batch-processing {
|
||||
&.el-dropdown-menu__item--divided {
|
||||
margin-top: 0;
|
||||
border-top: none;
|
||||
color: #909399;
|
||||
cursor: auto;
|
||||
font-size: 12px;
|
||||
line-height: 30px;
|
||||
border-bottom: 1px solid #E4E7ED;
|
||||
&:before {
|
||||
height: 0;
|
||||
}
|
||||
}
|
||||
&.el-dropdown-menu__item:not(.is-disabled):hover {
|
||||
color: #909399;
|
||||
background-color: #FFFFFF;
|
||||
}
|
||||
}
|
||||
</style>
|
||||
|
||||
@@ -43,16 +43,25 @@
|
||||
<template v-for="opt in options">
|
||||
<el-option
|
||||
v-if="data.type === 'select'"
|
||||
:key="opt.value"
|
||||
v-bind="opt"
|
||||
/>
|
||||
<!-- TODO: 支持 el-checkbox-button 变体 -->
|
||||
<el-checkbox
|
||||
v-else-if="data.type === 'checkbox-group'"
|
||||
:key="opt.label"
|
||||
v-bind="opt"
|
||||
/>
|
||||
<el-checkbox-button
|
||||
v-else-if="data.type === 'checkbox-group' && data.style === 'button'"
|
||||
:key="opt.value"
|
||||
v-bind="opt"
|
||||
:label="'value' in opt ? opt.value : opt.label"
|
||||
>
|
||||
{{ opt.value }}
|
||||
{{ opt.label }}
|
||||
</el-checkbox-button>
|
||||
|
||||
<el-checkbox
|
||||
v-else-if="data.type === 'checkbox-group' && data.style !== 'button'"
|
||||
:key="opt.value"
|
||||
v-bind="opt"
|
||||
:label="'value' in opt ? opt.value : opt.label"
|
||||
>
|
||||
{{ opt.label }}
|
||||
</el-checkbox>
|
||||
<!-- WARNING: radio 用 label 属性来表示 value 的含义 -->
|
||||
<!-- FYI: radio 的 value 属性可以在没有 radio-group 时用来关联到同一个 v-model -->
|
||||
@@ -229,7 +238,9 @@ export default {
|
||||
.then(resp => {
|
||||
if (isOptionsCase) {
|
||||
let formRenderer = this.$parent
|
||||
while (formRenderer.$options._componentTag !== 'el-form-renderer') { formRenderer = formRenderer.$parent }
|
||||
while (formRenderer.$options._componentTag !== 'el-form-renderer') {
|
||||
formRenderer = formRenderer.$parent
|
||||
}
|
||||
formRenderer.setOptions(this.prop, resp)
|
||||
} else {
|
||||
this.propsInner = { [prop]: resp }
|
||||
|
||||
@@ -14,10 +14,36 @@
|
||||
<slot v-for="item in fields" :slot="`$id:${item.id}`" :name="`$id:${item.id}`" />
|
||||
|
||||
<el-form-item v-if="hasButtons" class="form-buttons">
|
||||
<el-button v-for="button in moreButtons" :key="button.title" size="small" v-bind="button" :loading="button.loading" @click="handleClick(button)">{{ button.title }}</el-button>
|
||||
<el-button v-if="defaultButton && hasReset" size="small" @click="resetForm('form')">{{ $t('common.Reset') }}</el-button>
|
||||
<el-button v-if="defaultButton && hasSaveContinue" size="small" @click="submitForm('form', true)">{{ $t('common.SaveAndAddAnother') }}</el-button>
|
||||
<el-button v-if="defaultButton" size="small" :loading="isSubmitting" type="primary" @click="submitForm('form')">{{ $t('common.Submit') }}</el-button>
|
||||
<el-button
|
||||
v-for="button in moreButtons"
|
||||
:key="button.title"
|
||||
:loading="button.loading"
|
||||
size="small"
|
||||
v-bind="button"
|
||||
@click="handleClick(button)"
|
||||
>
|
||||
{{ button.title }}
|
||||
</el-button>
|
||||
<el-button v-if="defaultButton && hasReset" size="small" @click="resetForm('form')">
|
||||
{{ $t('common.Reset') }}
|
||||
</el-button>
|
||||
<el-button
|
||||
v-if="defaultButton && hasSaveContinue"
|
||||
size="small"
|
||||
@click="submitForm('form', true)"
|
||||
>
|
||||
{{ $t('common.SaveAndAddAnother') }}
|
||||
</el-button>
|
||||
<el-button
|
||||
v-if="defaultButton"
|
||||
:disabled="!canSubmit"
|
||||
:loading="isSubmitting"
|
||||
size="small"
|
||||
type="primary"
|
||||
@click="submitForm('form')"
|
||||
>
|
||||
{{ $t('common.Submit') }}
|
||||
</el-button>
|
||||
</el-form-item>
|
||||
</ElFormRender>
|
||||
</template>
|
||||
@@ -25,6 +51,7 @@
|
||||
<script>
|
||||
import ElFormRender from './components/el-form-renderer'
|
||||
import { scrollToError } from '@/utils'
|
||||
|
||||
export default {
|
||||
components: {
|
||||
ElFormRender
|
||||
@@ -42,6 +69,10 @@ export default {
|
||||
type: Boolean,
|
||||
default: true
|
||||
},
|
||||
canSubmit: {
|
||||
type: Boolean,
|
||||
default: true
|
||||
},
|
||||
hasSaveContinue: {
|
||||
type: Boolean,
|
||||
default: true
|
||||
@@ -70,6 +101,9 @@ export default {
|
||||
}
|
||||
},
|
||||
computed: {
|
||||
elForm() {
|
||||
return this.$refs.form
|
||||
},
|
||||
mobile() {
|
||||
return this.$store.state.app.device === 'mobile'
|
||||
},
|
||||
@@ -102,6 +136,8 @@ export default {
|
||||
const form = this.$refs['form']
|
||||
const values = form.getFormValue()
|
||||
callback(values, form, button)
|
||||
},
|
||||
getFormValue() {
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -125,7 +161,7 @@ export default {
|
||||
}
|
||||
|
||||
.el-form ::v-deep .el-form-item__error {
|
||||
position: inherit;
|
||||
position: inherit;
|
||||
}
|
||||
|
||||
.el-form ::v-deep .form-group-header {
|
||||
@@ -144,10 +180,12 @@ export default {
|
||||
font-size: 12px;
|
||||
line-height: 18px;
|
||||
}
|
||||
|
||||
.el-form ::v-deep .help-block a {
|
||||
color: var(--color-primary);
|
||||
}
|
||||
|
||||
.form-buttons {
|
||||
padding-top: 10px;
|
||||
margin-top: 20px;
|
||||
}
|
||||
</style>
|
||||
|
||||
@@ -14,9 +14,23 @@ export const EmailCheck = {
|
||||
trigger: ['blur', 'change']
|
||||
}
|
||||
|
||||
export const IpCheck = {
|
||||
required: true,
|
||||
validator: (rule, value, callback) => {
|
||||
value = value?.trim()
|
||||
const urlRegExp = /^[\w://.?=&#-]+$/
|
||||
if (urlRegExp.test(value)) {
|
||||
callback()
|
||||
} else {
|
||||
callback(new Error(i18n.t('common.FormatError')))
|
||||
}
|
||||
},
|
||||
trigger: ['blur', 'change']
|
||||
}
|
||||
|
||||
export const specialEmojiCheck = {
|
||||
validator: (rule, value, callback) => {
|
||||
value = value.trim()
|
||||
value = value?.trim()
|
||||
if (/[\uD83C[\uDF00-\uDFFF]|\uD83D[\uDC00-\uDE4F]/.test(value)) {
|
||||
callback(new Error(i18n.t('common.NotSpecialEmoji')))
|
||||
} else {
|
||||
@@ -26,11 +40,26 @@ export const specialEmojiCheck = {
|
||||
trigger: ['blur', 'change']
|
||||
}
|
||||
|
||||
// 只能输入字母、数字、下划线
|
||||
export const matchAlphanumericUnderscore = {
|
||||
validator: (rule, value, callback) => {
|
||||
value = value?.trim()
|
||||
if (!/^[a-zA-Z0-9_]+$/.test(value)) {
|
||||
callback(new Error(i18n.t('common.notAlphanumericUnderscore')))
|
||||
} else {
|
||||
callback()
|
||||
}
|
||||
},
|
||||
trigger: ['blur', 'change']
|
||||
}
|
||||
|
||||
export default {
|
||||
IpCheck,
|
||||
Required,
|
||||
RequiredChange,
|
||||
EmailCheck,
|
||||
specialEmojiCheck
|
||||
specialEmojiCheck,
|
||||
matchAlphanumericUnderscore
|
||||
}
|
||||
|
||||
export const JsonRequired = {
|
||||
@@ -38,7 +67,7 @@ export const JsonRequired = {
|
||||
trigger: 'change',
|
||||
validator: (rule, value, callback) => {
|
||||
try {
|
||||
JSON.parse(value)
|
||||
typeof value === 'string' ? JSON.parse(value) : value
|
||||
callback()
|
||||
} catch (e) {
|
||||
callback(new Error(i18n.t('common.InvalidJson')))
|
||||
@@ -51,8 +80,8 @@ export const JsonRequiredUserNameMapped = {
|
||||
trigger: 'change',
|
||||
validator: (rule, value, callback) => {
|
||||
try {
|
||||
JSON.parse(value)
|
||||
const hasUserName = _.map(JSON.parse(value), (value) => value)
|
||||
const v = typeof value === 'string' ? JSON.parse(value) : value
|
||||
const hasUserName = _.map(v, (value) => value)
|
||||
if (!hasUserName.includes('username')) {
|
||||
callback(new Error(i18n.t('common.requiredHasUserNameMapped')))
|
||||
}
|
||||
|
||||
@@ -8,12 +8,12 @@
|
||||
<el-table
|
||||
ref="table"
|
||||
v-loading="loading"
|
||||
v-bind="tableAttrs"
|
||||
:data="data"
|
||||
:row-class-name="rowClassName"
|
||||
v-bind="tableAttrs"
|
||||
@select="selectStrategy.onSelect"
|
||||
v-on="$listeners"
|
||||
@selection-change="selectStrategy.onSelectionChange"
|
||||
@select="selectStrategy.onSelect"
|
||||
@select-all="selectStrategy.onSelectAll($event, canSelect)"
|
||||
@sort-change="onSortChange"
|
||||
>
|
||||
@@ -91,28 +91,28 @@
|
||||
|
||||
<!--非树-->
|
||||
<template v-else>
|
||||
<el-data-table-column v-if="hasSelection" type="selection" :align="selectionAlign" :selectable="canSelect" />
|
||||
<el-data-table-column v-if="hasSelection" :align="selectionAlign" :selectable="canSelect" type="selection" />
|
||||
<el-data-table-column
|
||||
v-for="col in columns"
|
||||
:key="col.prop"
|
||||
:formatter="typeof col.formatter === 'function' ? col.formatter : null"
|
||||
:filters="col.filters || null"
|
||||
:filter-multiple="false"
|
||||
:filter-method="typeof col.filterMethod === 'function' ? col.filterMethod : null"
|
||||
:filter-multiple="false"
|
||||
:filters="col.filters || null"
|
||||
:formatter="typeof col.formatter === 'function' ? col.formatter : null"
|
||||
v-bind="{align: columnsAlign, ...col}"
|
||||
>
|
||||
<template v-if="col.formatter && typeof col.formatter !== 'function'" v-slot:default="{row, column, index}">
|
||||
<div
|
||||
:is="col.formatter"
|
||||
:key="row.id"
|
||||
:table-data="data"
|
||||
:row="row"
|
||||
:cell-value="row[col.prop]"
|
||||
:col="col"
|
||||
:column="column"
|
||||
:index="index"
|
||||
:url="url"
|
||||
:reload="getList"
|
||||
:col="col"
|
||||
:cell-value="row[col.prop]"
|
||||
:row="row"
|
||||
:table-data="data"
|
||||
:url="url"
|
||||
/>
|
||||
</template>
|
||||
</el-data-table-column>
|
||||
@@ -122,13 +122,12 @@
|
||||
|
||||
<el-pagination
|
||||
v-if="hasPagination"
|
||||
:current-page="page"
|
||||
:page-sizes="paginationSizes"
|
||||
:page-size="size"
|
||||
:total="total"
|
||||
:background="paginationBackground"
|
||||
style="text-align: right; padding: 10px 0;"
|
||||
:current-page="page"
|
||||
:layout="paginationLayout"
|
||||
:page-size="size"
|
||||
:page-sizes="paginationSizes"
|
||||
:total="total"
|
||||
v-bind="extraPaginationAttrs"
|
||||
@size-change="handleSizeChange"
|
||||
@current-change="handleCurrentChange"
|
||||
@@ -136,18 +135,18 @@
|
||||
|
||||
<the-dialog
|
||||
ref="dialog"
|
||||
:new-title="dialogNewTitle"
|
||||
:button-size="buttonSize"
|
||||
:dialog-attrs="dialogAttrs"
|
||||
:edit-title="dialogEditTitle"
|
||||
:view-title="dialogViewTitle"
|
||||
:form="form"
|
||||
:form-attrs="formAttrs"
|
||||
:dialog-attrs="dialogAttrs"
|
||||
:button-size="buttonSize"
|
||||
:new-title="dialogNewTitle"
|
||||
:view-title="dialogViewTitle"
|
||||
@confirm="onConfirm"
|
||||
>
|
||||
<template v-slot="scope">
|
||||
<!-- @slot 表单作用域插槽。当编辑、查看时传入row;新增时row=null -->
|
||||
<slot name="form" :row="scope.row" />
|
||||
<slot :row="scope.row" name="form" />
|
||||
</template>
|
||||
</the-dialog>
|
||||
</template>
|
||||
@@ -167,6 +166,7 @@ import getLocatedSlotKeys from './utils/extract-keys'
|
||||
import transformSearchImmediatelyItem from './utils/search-immediately-item'
|
||||
import isFalsey from './utils/is-falsey'
|
||||
import merge from 'deepmerge'
|
||||
|
||||
const defaultFirstPage = 1
|
||||
const noPaginationDataPath = 'payload'
|
||||
|
||||
@@ -268,7 +268,8 @@ export default {
|
||||
*/
|
||||
beforeSearch: {
|
||||
type: Function,
|
||||
default() {}
|
||||
default() {
|
||||
}
|
||||
},
|
||||
/**
|
||||
* 单选, 适用场景: 不可以批量删除
|
||||
@@ -360,28 +361,36 @@ export default {
|
||||
*/
|
||||
newText: {
|
||||
type: String,
|
||||
default: '新增'
|
||||
default: function() {
|
||||
return this.$t('ops.Add')
|
||||
}
|
||||
},
|
||||
/**
|
||||
* 修改按钮文案
|
||||
*/
|
||||
editText: {
|
||||
type: String,
|
||||
default: '修改'
|
||||
default: function() {
|
||||
return this.$t('ops.Modify')
|
||||
}
|
||||
},
|
||||
/**
|
||||
* 查看按钮文案
|
||||
*/
|
||||
viewText: {
|
||||
type: String,
|
||||
default: '查看'
|
||||
default: function() {
|
||||
return this.$t('ops.View')
|
||||
}
|
||||
},
|
||||
/**
|
||||
* 删除按钮文案
|
||||
*/
|
||||
deleteText: {
|
||||
type: String,
|
||||
default: '删除'
|
||||
default: function() {
|
||||
return this.$t('ops.Delete')
|
||||
}
|
||||
},
|
||||
/**
|
||||
* 删除提示语。接受要删除的数据(单个对象或数组);返回字符串
|
||||
@@ -391,7 +400,7 @@ export default {
|
||||
deleteMessage: {
|
||||
type: Function,
|
||||
default() {
|
||||
return `确认${this.deleteText}吗?`
|
||||
return this.$t('ops.Confirm') + this.deleteText + '?'
|
||||
}
|
||||
},
|
||||
/**
|
||||
@@ -450,7 +459,7 @@ export default {
|
||||
onSuccess: {
|
||||
type: Function,
|
||||
default() {
|
||||
return this.$message.success('操作成功')
|
||||
return this.$message.success(this.$t('ops.SuccessfulOperation'))
|
||||
}
|
||||
},
|
||||
/**
|
||||
@@ -707,7 +716,8 @@ export default {
|
||||
},
|
||||
extraPaginationAttrs: {
|
||||
type: Object,
|
||||
default: () => {}
|
||||
default: () => {
|
||||
}
|
||||
},
|
||||
hasSelection: {
|
||||
type: Boolean,
|
||||
@@ -1037,6 +1047,7 @@ export default {
|
||||
} else {
|
||||
this.innerQuery = merge(this.innerQuery, attrs)
|
||||
}
|
||||
this.selected.splice(0, this.selected.length)
|
||||
return this.getList()
|
||||
},
|
||||
searchDate(attrs) {
|
||||
@@ -1156,7 +1167,7 @@ export default {
|
||||
* @param {object|object[]} - 要删除的数据对象或数组
|
||||
*/
|
||||
onDefaultDelete(data) {
|
||||
this.$confirm(this.deleteMessage(data), '提示', {
|
||||
this.$confirm(this.deleteMessage(data), this.$t('common.Info'), {
|
||||
type: 'warning',
|
||||
confirmButtonClass: 'el-button--danger',
|
||||
beforeClose: async(action, instance, done) => {
|
||||
@@ -1199,7 +1210,9 @@ export default {
|
||||
remain === 0 &&
|
||||
this.page === lastPage &&
|
||||
this.page > defaultFirstPage
|
||||
) { this.page-- }
|
||||
) {
|
||||
this.page--
|
||||
}
|
||||
},
|
||||
|
||||
// 树形table相关
|
||||
|
||||
@@ -1,16 +1,22 @@
|
||||
.el-data-table ::v-deep .el-pagination{
|
||||
text-align: center !important;
|
||||
}
|
||||
.el-data-table ::v-deep .el-table td{
|
||||
padding: 4px 0;
|
||||
}
|
||||
.el-data-table ::v-deep .el-table th{
|
||||
padding: 4px 0;
|
||||
}
|
||||
.el-data-table ::v-deep .el-form-item{
|
||||
margin-bottom:10px !important ;
|
||||
margin-top:10px;
|
||||
}
|
||||
.el-data-table ::v-deep .el-pagination{
|
||||
padding:15px 0 !important ;
|
||||
.el-data-table ::v-deep .el-table td {
|
||||
padding: 4px 0;
|
||||
}
|
||||
|
||||
.el-data-table ::v-deep .el-table th {
|
||||
padding: 4px 0;
|
||||
}
|
||||
|
||||
.el-data-table ::v-deep .el-form-item {
|
||||
margin-bottom: 10px !important;
|
||||
margin-top: 10px;
|
||||
}
|
||||
|
||||
.el-data-table ::v-deep .el-pagination {
|
||||
text-align: right;
|
||||
padding: 15px 25px 20px 15px;
|
||||
|
||||
.el-pagination__total {
|
||||
float: left;
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
@@ -1,11 +1,11 @@
|
||||
<template>
|
||||
<ElDatableTable
|
||||
ref="table"
|
||||
class="el-table"
|
||||
class="el-data-table"
|
||||
v-bind="tableConfig"
|
||||
@sizeChange="handleSizeChange"
|
||||
@update="onUpdate"
|
||||
v-on="iListeners"
|
||||
@sizeChange="handleSizeChange"
|
||||
/>
|
||||
</template>
|
||||
|
||||
@@ -21,7 +21,8 @@ export default {
|
||||
props: {
|
||||
config: {
|
||||
type: Object,
|
||||
default: () => {}
|
||||
default: () => {
|
||||
}
|
||||
}
|
||||
},
|
||||
data() {
|
||||
@@ -49,7 +50,11 @@ export default {
|
||||
stripe: false, // 斑马纹表格
|
||||
border: true, // 表格边框
|
||||
fit: true, // 宽度自适应,
|
||||
tooltipEffect: 'dark'
|
||||
tooltipEffect: 'dark',
|
||||
rowClassName: ({ row }) => {
|
||||
const selected = this.dataTable.selected.find(item => item.id === row.id)
|
||||
return selected ? 'selected-row' : ''
|
||||
}
|
||||
},
|
||||
extraButtons: userTableActions.extraButtons,
|
||||
onEdit: (row) => {
|
||||
@@ -65,6 +70,7 @@ export default {
|
||||
},
|
||||
pageCount: 5,
|
||||
paginationLayout: 'total, sizes, prev, pager, next',
|
||||
paginationSize: JSON.parse(localStorage.getItem('paginationSize')) || 15,
|
||||
paginationSizes: [15, 30, 50, 100],
|
||||
paginationBackground: true,
|
||||
transformQuery: query => {
|
||||
@@ -86,40 +92,36 @@ export default {
|
||||
}
|
||||
return query
|
||||
},
|
||||
theRowDefaultIsSelected: (row) => { return false }
|
||||
theRowDefaultIsSelected: (row) => {
|
||||
return false
|
||||
}
|
||||
}
|
||||
}
|
||||
},
|
||||
computed: {
|
||||
iListeners() {
|
||||
const defaultListeners = {}
|
||||
return Object.assign(defaultListeners, this.$listeners, this.tableConfig?.listeners)
|
||||
},
|
||||
dataTable() {
|
||||
return this.$refs.table
|
||||
},
|
||||
tableConfig() {
|
||||
const tableDefaultConfig = this.defaultConfig
|
||||
tableDefaultConfig.paginationSize = _.get(this.globalTableConfig, 'paginationSize', 15)
|
||||
let tableAttrs = tableDefaultConfig.tableAttrs
|
||||
if (this.config.tableAttrs) {
|
||||
tableAttrs = Object.assign(tableAttrs, this.config.tableAttrs)
|
||||
}
|
||||
const config = Object.assign(tableDefaultConfig, this.config)
|
||||
config.tableAttrs = tableAttrs
|
||||
this.$log.debug('elTableConfig', config)
|
||||
return config
|
||||
},
|
||||
iListeners() {
|
||||
return Object.assign({}, this.$listeners, this.tableConfig.listeners)
|
||||
},
|
||||
dataTable() {
|
||||
return this.$refs.table
|
||||
},
|
||||
...mapGetters({
|
||||
'globalTableConfig': 'tableConfig'
|
||||
})
|
||||
},
|
||||
watch: {
|
||||
config: {
|
||||
handler() {
|
||||
// this.getList()
|
||||
},
|
||||
deep: true
|
||||
}
|
||||
},
|
||||
watch: {},
|
||||
methods: {
|
||||
getList() {
|
||||
this.$refs.table.clearSelection()
|
||||
@@ -153,6 +155,7 @@ export default {
|
||||
}
|
||||
},
|
||||
handleSizeChange(val) {
|
||||
localStorage.setItem('paginationSize', val)
|
||||
this.$store.commit('table/SET_TABLE_CONFIG',
|
||||
{
|
||||
key: 'paginationSize',
|
||||
@@ -164,37 +167,5 @@ export default {
|
||||
}
|
||||
</script>
|
||||
|
||||
<style lang="less" scoped>
|
||||
|
||||
.el-table ::v-deep .el-table__row > td {
|
||||
line-height: 1.5;
|
||||
padding: 8px 0;
|
||||
}
|
||||
.el-table ::v-deep .el-table__row > td> div > span {
|
||||
text-overflow: ellipsis;
|
||||
overflow: hidden;
|
||||
white-space: nowrap;
|
||||
}
|
||||
.el-table ::v-deep .el-table__header > thead > tr >th {
|
||||
padding: 8px 0;
|
||||
background-color: #F5F5F6;
|
||||
font-size: 13px;
|
||||
line-height: 1.5;
|
||||
}
|
||||
.table{
|
||||
margin-top: 15px;
|
||||
}
|
||||
|
||||
//分页
|
||||
.el-pagination ::v-deep .el-pagination__total{
|
||||
float: left;
|
||||
}
|
||||
|
||||
.el-pagination ::v-deep .el-pagination__sizes{
|
||||
float: left;
|
||||
}
|
||||
//修改颜色
|
||||
// .el-button--text{
|
||||
// color: #409EFF;
|
||||
// }
|
||||
<style lang="scss" scoped>
|
||||
</style>
|
||||
|
||||
@@ -1,47 +1,31 @@
|
||||
<template>
|
||||
<div>
|
||||
<div
|
||||
v-if="treeSetting.customTreeHeader"
|
||||
class="tree-header treebox"
|
||||
>
|
||||
<div class="content">
|
||||
<span class="title">
|
||||
{{ treeSetting.customTreeHeaderName }}
|
||||
</span>
|
||||
<span class="tree-banner-icon-zone">
|
||||
<a id="searchIcon" class="tree-search special">
|
||||
<i
|
||||
class="fa fa-search tree-banner-icon"
|
||||
@click.stop="treeSearch"
|
||||
<div class="treebox">
|
||||
<div>
|
||||
<el-input
|
||||
v-if="treeSetting.showSearch && showTreeSearch"
|
||||
v-model="treeSearchValue"
|
||||
:placeholder="$tc('common.Search')"
|
||||
class="fixed-tree-search"
|
||||
prefix-icon="fa fa-search"
|
||||
size="mini"
|
||||
@input="treeSearchHandle"
|
||||
>
|
||||
<span slot="suffix">
|
||||
<!-- <i class="fa fa-search" style="font-size: 14px; color: #676A6C;" /> -->
|
||||
<svg-icon
|
||||
:icon-class="'close'"
|
||||
class="icon"
|
||||
style="font-size: 14px;"
|
||||
@click="onClose"
|
||||
/>
|
||||
<input
|
||||
id="searchInput"
|
||||
v-model="treeSearchValue"
|
||||
type="text"
|
||||
autocomplete="off"
|
||||
class="tree-input"
|
||||
>
|
||||
</a>
|
||||
<i
|
||||
class="fa fa-refresh tree-banner-icon"
|
||||
style="margin-right: 2px;"
|
||||
@click.stop="refresh"
|
||||
/>
|
||||
</span>
|
||||
</span>
|
||||
</el-input>
|
||||
</div>
|
||||
<ul v-show="loading" class="ztree">
|
||||
{{ this.$t('common.tree.Loading') }}...
|
||||
</ul>
|
||||
<ul v-show="!loading" :id="iZTreeID" class="ztree" />
|
||||
<div v-if="treeSetting.treeUrl===''" class="tree-empty">
|
||||
{{ this.$t('common.tree.Empty') }}
|
||||
</div>
|
||||
</div>
|
||||
<div v-else class="treebox">
|
||||
<ul v-show="loading" class="ztree">
|
||||
{{ this.$t('common.tree.Loading') }}...
|
||||
</ul>
|
||||
<ul v-show="!loading" :id="iZTreeID" class="ztree" />
|
||||
<ul v-show="!loading" :id="iZTreeID" :key="iZTreeID" class="ztree" />
|
||||
<div v-if="treeSetting.treeUrl===''" class="tree-empty">
|
||||
{{ this.$t('common.tree.Empty') }}
|
||||
<a id="tree-refresh"><i class="fa fa-refresh" /></a>
|
||||
@@ -62,16 +46,17 @@ import $ from '@/utils/jquery-vendor.js'
|
||||
import '@ztree/ztree_v3/js/jquery.ztree.all.min.js'
|
||||
import '@ztree/ztree_v3/js/jquery.ztree.exhide.min.js'
|
||||
import '@/styles/ztree.css'
|
||||
import '@/styles/ztree_icon.css'
|
||||
import axiosRetry from 'axios-retry'
|
||||
|
||||
const defaultObject = {
|
||||
type: Object,
|
||||
default: () => {}
|
||||
default: () => {
|
||||
}
|
||||
}
|
||||
export default {
|
||||
name: 'ZTree',
|
||||
components: {
|
||||
},
|
||||
components: {},
|
||||
props: {
|
||||
setting: defaultObject
|
||||
},
|
||||
@@ -83,6 +68,7 @@ export default {
|
||||
rMenu: '',
|
||||
init: false,
|
||||
loading: false,
|
||||
showTreeSearch: JSON.parse(localStorage.getItem('showTreeSearch')) || false,
|
||||
treeSearchValue: ''
|
||||
}
|
||||
},
|
||||
@@ -93,88 +79,99 @@ export default {
|
||||
},
|
||||
mounted() {
|
||||
window.refresh = this.refresh
|
||||
window.treeSearch = this.treeSearch
|
||||
window.onSearch = this.onSearch
|
||||
this.initTree()
|
||||
},
|
||||
beforeDestroy() {
|
||||
$.fn.zTree.destroy(this.iZTreeID)
|
||||
},
|
||||
methods: {
|
||||
initTree: function() {
|
||||
async initTree(refresh = false) {
|
||||
const vm = this
|
||||
let treeUrl
|
||||
if (this.init) {
|
||||
this.loading = true
|
||||
}
|
||||
if (this.init && this.treeSetting.treeUrl.indexOf('/perms/') !== -1 && this.treeSetting.treeUrl.indexOf('rebuild_tree') === -1) {
|
||||
treeUrl = (this.treeSetting.treeUrl.indexOf('?') === -1) ? `${this.treeSetting.treeUrl}?rebuild_tree=1` : `${this.treeSetting.treeUrl}&rebuild_tree=1`
|
||||
this.loading = true
|
||||
if (refresh && this.treeSetting.treeUrl.indexOf('/perms/') !== -1 &&
|
||||
this.treeSetting.treeUrl.indexOf('rebuild_tree') === -1
|
||||
) {
|
||||
treeUrl = (this.treeSetting.treeUrl.indexOf('?') === -1)
|
||||
? `${this.treeSetting.treeUrl}?rebuild_tree=1`
|
||||
: `${this.treeSetting.treeUrl}&rebuild_tree=1`
|
||||
} else {
|
||||
treeUrl = this.treeSetting.treeUrl
|
||||
}
|
||||
this.$axios.get(treeUrl, {
|
||||
|
||||
if (refresh) {
|
||||
$.fn.zTree.destroy(this.iZTreeID)
|
||||
}
|
||||
|
||||
let res = await this.$axios.get(treeUrl, {
|
||||
'axios-retry': {
|
||||
retries: 20,
|
||||
retryCondition: e => {
|
||||
return axiosRetry.isNetworkOrIdempotentRequestError(e) || e.response.status === 409
|
||||
},
|
||||
shouldResetTimeout: true,
|
||||
retryDelay: () => { return 5000 }
|
||||
retryDelay: () => {
|
||||
return 5000
|
||||
}
|
||||
}
|
||||
}).then(res => {
|
||||
if (!res) res = []
|
||||
if (res.length === 0) {
|
||||
res.push({
|
||||
name: this.$t('common.tree.Empty')
|
||||
})
|
||||
}
|
||||
this.treeSetting.treeUrl = treeUrl
|
||||
if (this.init) {
|
||||
vm.zTree.destroy()
|
||||
}
|
||||
|
||||
this.zTree = $.fn.zTree.init($(`#${this.iZTreeID}`), this.treeSetting, res)
|
||||
if (!this.treeSetting.customTreeHeader) {
|
||||
this.rootNodeAddDom(this.zTree)
|
||||
}
|
||||
// 手动上报事件, Tree加载完成
|
||||
this.$emit('TreeInitFinish', this.zTree)
|
||||
|
||||
if (this.treeSetting.showMenu) {
|
||||
this.rMenu = $(`#${this.iRMenuID}`)
|
||||
}
|
||||
if (this.treeSetting.otherMenu) {
|
||||
$('.menu-actions').append(this.otherMenu)
|
||||
}
|
||||
}).finally(_ => {
|
||||
vm.loading = false
|
||||
vm.init = true
|
||||
})
|
||||
vm.loading = false
|
||||
if (!res) res = []
|
||||
if (res?.length === 0) {
|
||||
res?.push({
|
||||
name: this.$t('common.tree.Empty')
|
||||
})
|
||||
}
|
||||
this.treeSetting.treeUrl = treeUrl
|
||||
vm.zTree = $.fn.zTree.init($(`#${this.iZTreeID}`), this.treeSetting, res)
|
||||
const rootNode = this.zTree.getNodes()[0]
|
||||
this.rootNodeAddDom(rootNode)
|
||||
// 手动上报事件, Tree加载完成
|
||||
this.$emit('TreeInitFinish', this.zTree)
|
||||
|
||||
if (this.treeSetting.showMenu) {
|
||||
this.rMenu = $(`#${this.iRMenuID}`)
|
||||
}
|
||||
if (this.treeSetting?.otherMenu) {
|
||||
$('.menu-actions').append(this.otherMenu)
|
||||
}
|
||||
},
|
||||
rootNodeAddDom(ztree) {
|
||||
onSearch() {
|
||||
this.showTreeSearch = !this.showTreeSearch
|
||||
localStorage.setItem('showTreeSearch', JSON.stringify(this.showTreeSearch))
|
||||
},
|
||||
onClose() {
|
||||
this.refresh()
|
||||
this.onSearch()
|
||||
},
|
||||
rootNodeAddDom(rootNode) {
|
||||
const { showSearch, showRefresh } = this.treeSetting
|
||||
const searchIcon = `<a class="tree-search" id="searchIcon">
|
||||
<i class='fa fa-search tree-banner-icon' onclick="treeSearch()" /></i>
|
||||
<input type="text" autocomplete="off" id="searchInput" class="tree-input" />
|
||||
</a>`
|
||||
const refreshIcon = "<a id='tree-refresh' onclick='refresh()'><i class='fa fa-refresh'></i></a>"
|
||||
const searchIcon = `
|
||||
<a class="tree-action-btn" id="search-btn" onclick="onSearch()">
|
||||
<i class="fa fa-search tree-banner-icon"></i>
|
||||
</a>`
|
||||
const refreshIcon = `
|
||||
<a id="tree-refresh" class="tree-action-btn" onclick="refresh()">
|
||||
<i class="fa fa-refresh"></i>
|
||||
</a>`
|
||||
const treeActions = `${showSearch ? searchIcon : ''}${showRefresh ? refreshIcon : ''}`
|
||||
const icons = `<span class="">${treeActions}</span>`
|
||||
const rootNode = ztree.getNodes()[0]
|
||||
const icons = `
|
||||
<span style="float: right; margin-right: 10px">
|
||||
${treeActions}
|
||||
</span>`
|
||||
if (rootNode) {
|
||||
const $rootNodeRef = $('#' + rootNode.tId + '_a')
|
||||
$rootNodeRef.after(icons)
|
||||
}
|
||||
},
|
||||
refresh() {
|
||||
async refresh() {
|
||||
this.treeSearchValue = ''
|
||||
const result = this.treeSetting?.callback?.refresh()
|
||||
if (result && result.then) {
|
||||
result.finally(() => {
|
||||
this.initTree()
|
||||
})
|
||||
} else {
|
||||
this.initTree()
|
||||
if (this.treeSetting?.callback?.refresh) {
|
||||
await this.treeSetting.callback.refresh()
|
||||
}
|
||||
this.zTree.destroy()
|
||||
setTimeout(() => this.initTree(true), 200)
|
||||
},
|
||||
treeSearch() {
|
||||
const searchIcon = document.getElementById(`searchIcon`)
|
||||
@@ -190,16 +187,15 @@ export default {
|
||||
searchIcon.classList.toggle('active')
|
||||
}
|
||||
}
|
||||
searchInput.oninput = _.debounce((e) => {
|
||||
e.stopPropagation()
|
||||
const value = e.target.value || ''
|
||||
if (this.treeSetting.async.enable) {
|
||||
this.filterAssetsServer(value)
|
||||
} else {
|
||||
this.filterTree(value)
|
||||
}
|
||||
}, 600)
|
||||
searchInput.oninput = e => this.treeSearchHandle((e.target.value || ''))
|
||||
},
|
||||
treeSearchHandle: _.debounce(function(value) {
|
||||
if (this.treeSetting.async.enable) {
|
||||
this.filterAssetsServer(value)
|
||||
} else {
|
||||
this.filterTree(value)
|
||||
}
|
||||
}, 600),
|
||||
getCheckedNodes: function() {
|
||||
return this.zTree.getCheckedNodes(true)
|
||||
},
|
||||
@@ -310,7 +306,7 @@ export default {
|
||||
this.zTree.hideNodes(treeNodes)
|
||||
}
|
||||
|
||||
let treeUrl = this.treeSetting.treeUrl
|
||||
let treeUrl = this.treeSetting.searchUrl ? this.treeSetting.searchUrl : this.treeSetting.treeUrl
|
||||
const filterField = treeUrl.includes('?') ? `&search=${keyword}` : `?search=${keyword}`
|
||||
if (treeUrl.indexOf('assets/nodes/children/tree') > -1) {
|
||||
treeUrl = treeUrl + '&all=all'
|
||||
@@ -323,8 +319,10 @@ export default {
|
||||
const newNode = { id: 'search', name: name, isParent: true, open: true, zAsync: true }
|
||||
searchNode = this.zTree.addNodes(null, newNode)[0]
|
||||
searchNode.zAsync = true
|
||||
this.rootNodeAddDom(searchNode)
|
||||
|
||||
const nodesGroupByOrg = this.groupBy(nodes, (node) => {
|
||||
return node.meta.data.org_name
|
||||
return node.meta?.data?.org_name
|
||||
})
|
||||
|
||||
for (const item of nodesGroupByOrg) {
|
||||
@@ -332,180 +330,267 @@ export default {
|
||||
}
|
||||
searchNode.open = true
|
||||
})
|
||||
return
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
</script>
|
||||
|
||||
<style lang='scss' scoped>
|
||||
div.rMenu {
|
||||
position: absolute;
|
||||
visibility: hidden;
|
||||
text-align: left;
|
||||
top: 0;
|
||||
left: 0;
|
||||
z-index: 999;
|
||||
float: left;
|
||||
padding: 0 0;
|
||||
margin: 2px 0 0;
|
||||
list-style: none;
|
||||
background-clip: padding-box;
|
||||
}
|
||||
.dataTables_wrapper .dataTables_processing {
|
||||
opacity: .9;
|
||||
border: none;
|
||||
}
|
||||
div.rMenu li{
|
||||
margin: 6px 0;
|
||||
cursor: pointer;
|
||||
list-style: none outside none;
|
||||
}
|
||||
.dropdown-menu {
|
||||
border: medium none;
|
||||
min-width: 160px;
|
||||
background-color: #fff;
|
||||
border-radius: 3px;
|
||||
box-shadow: 0 0 3px rgba(86, 96, 117, 0.7);
|
||||
display: block;
|
||||
float: left;
|
||||
font-size: 12px;
|
||||
left: 0;
|
||||
list-style: none outside none;
|
||||
padding: 0;
|
||||
position: absolute;
|
||||
text-shadow: none;
|
||||
top: 100%;
|
||||
z-index: 1000;
|
||||
}
|
||||
.ztree ::v-deep .fa {
|
||||
font: normal normal normal 14px/1 FontAwesome !important;
|
||||
}
|
||||
.dropdown a:hover {
|
||||
background-color: #f1f1f1
|
||||
}
|
||||
.dropdown-menu > li > a {
|
||||
border-radius: 3px;
|
||||
color: inherit;
|
||||
line-height: 25px;
|
||||
margin: 4px;
|
||||
text-align: left;
|
||||
font-weight: normal;
|
||||
display: block;
|
||||
padding: 3px 20px;
|
||||
clear: both;
|
||||
white-space: nowrap;
|
||||
}
|
||||
.dropdown-menu>li>a:hover, .dropdown-menu>li>a:focus {
|
||||
color: #262626;
|
||||
text-decoration: none;
|
||||
background-color: #f5f5f5;
|
||||
}
|
||||
.treebox {
|
||||
height: 80vh;
|
||||
<style lang="scss" scoped>
|
||||
::-webkit-scrollbar-corner {
|
||||
background: transparent;
|
||||
}
|
||||
|
||||
::-webkit-scrollbar-track:horizontal {
|
||||
background: #FFFFFF;
|
||||
border-radius: 10px;
|
||||
}
|
||||
|
||||
div.rMenu {
|
||||
position: absolute;
|
||||
visibility: hidden;
|
||||
text-align: left;
|
||||
top: 0;
|
||||
left: 0;
|
||||
z-index: 999;
|
||||
float: left;
|
||||
padding: 0 0;
|
||||
margin: 2px 0 0;
|
||||
list-style: none;
|
||||
background-clip: padding-box;
|
||||
}
|
||||
|
||||
.dataTables_wrapper .dataTables_processing {
|
||||
opacity: .9;
|
||||
border: none;
|
||||
}
|
||||
|
||||
div.rMenu li {
|
||||
margin: 6px 0;
|
||||
cursor: pointer;
|
||||
list-style: none outside none;
|
||||
}
|
||||
|
||||
.dropdown-menu {
|
||||
border: medium none;
|
||||
min-width: 160px;
|
||||
background-color: #fff;
|
||||
border-radius: 3px;
|
||||
box-shadow: 0 0 3px rgba(86, 96, 117, 0.7);
|
||||
display: block;
|
||||
float: left;
|
||||
font-size: 12px;
|
||||
left: 0;
|
||||
list-style: none outside none;
|
||||
padding: 0;
|
||||
position: absolute;
|
||||
text-shadow: none;
|
||||
top: 100%;
|
||||
z-index: 1000;
|
||||
}
|
||||
|
||||
.ztree ::v-deep .fa {
|
||||
font: normal normal normal 14px/1 FontAwesome !important;
|
||||
}
|
||||
|
||||
.dropdown a:hover {
|
||||
background-color: #f1f1f1
|
||||
}
|
||||
|
||||
.dropdown-menu > li > a {
|
||||
border-radius: 3px;
|
||||
color: inherit;
|
||||
line-height: 25px;
|
||||
margin: 4px;
|
||||
text-align: left;
|
||||
font-weight: normal;
|
||||
display: block;
|
||||
padding: 3px 20px;
|
||||
clear: both;
|
||||
white-space: nowrap;
|
||||
}
|
||||
|
||||
.dropdown-menu > li > a:hover, .dropdown-menu > li > a:focus {
|
||||
color: #262626;
|
||||
text-decoration: none;
|
||||
background-color: #f5f5f5;
|
||||
}
|
||||
|
||||
.treebox {
|
||||
background-color: transparent;
|
||||
|
||||
> > > .ztree {
|
||||
overflow: auto;
|
||||
background-color: transparent;
|
||||
height: calc(100vh - 237px);
|
||||
|
||||
li {
|
||||
background-color: transparent !important;
|
||||
|
||||
.button {
|
||||
background-color: rgba(0, 0, 0, 0);
|
||||
}
|
||||
|
||||
ul {
|
||||
background-color: transparent !important;
|
||||
}
|
||||
}
|
||||
}
|
||||
::v-deep #tree-refresh {
|
||||
margin-left: 3px;
|
||||
}
|
||||
|
||||
::v-deep #tree-refresh {
|
||||
margin-left: 3px;
|
||||
}
|
||||
|
||||
::v-deep .tree-banner-icon-zone {
|
||||
position: absolute;
|
||||
right: 7px;
|
||||
height: 30px;
|
||||
overflow: hidden;
|
||||
|
||||
.fa {
|
||||
color: #838385 !important;;
|
||||
|
||||
&:hover {
|
||||
color: #606266 !important;;
|
||||
}
|
||||
}
|
||||
::v-deep .tree-banner-icon-zone {
|
||||
position: absolute;
|
||||
right: 7px;
|
||||
}
|
||||
|
||||
::v-deep .tree-search {
|
||||
position: relative;
|
||||
top: -2px;
|
||||
width: 20px;
|
||||
height: 20px;
|
||||
display: inline-block;
|
||||
border-radius: 12px;
|
||||
vertical-align: sub;
|
||||
transition: .25s;
|
||||
overflow: hidden;
|
||||
|
||||
.fa {
|
||||
width: 13px !important;
|
||||
}
|
||||
|
||||
.fa-search {
|
||||
padding-top: 1px;
|
||||
}
|
||||
}
|
||||
|
||||
::v-deep .tree-search .tree-banner-icon {
|
||||
position: absolute;
|
||||
top: 4px;
|
||||
left: 6px;
|
||||
border-radius: 12px;
|
||||
overflow: hidden;
|
||||
background-color: transparent !important;
|
||||
display: flex;
|
||||
justify-content: center;
|
||||
align-items: center;
|
||||
cursor: pointer;
|
||||
}
|
||||
|
||||
::v-deep .tree-search.active {
|
||||
width: 160px;
|
||||
background-color: #ffffff !important;
|
||||
}
|
||||
|
||||
::v-deep .tree-search.active:hover {
|
||||
border-radius: 12px;
|
||||
}
|
||||
|
||||
::v-deep .tree-search input {
|
||||
position: relative;
|
||||
left: 20px;
|
||||
width: 133px;
|
||||
height: 100%;
|
||||
background-color: #ffffff !important;
|
||||
color: #606266;
|
||||
display: flex;
|
||||
justify-content: center;
|
||||
align-items: center;
|
||||
border: none;
|
||||
outline: none;
|
||||
}
|
||||
|
||||
.tree-header {
|
||||
position: relative;
|
||||
|
||||
.title {
|
||||
font-weight: 500;
|
||||
}
|
||||
|
||||
.content {
|
||||
height: 30px;
|
||||
line-height: 30px;
|
||||
border-bottom: 1px solid #e0e0e0;
|
||||
border-radius: 3px;
|
||||
padding: 0 5px;
|
||||
box-sizing: border-box;
|
||||
overflow: hidden;
|
||||
.fa {
|
||||
color: #838385!important;;
|
||||
&:hover {
|
||||
color: #606266!important;;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
::v-deep .tree-search {
|
||||
position: relative;
|
||||
top: -2px;
|
||||
width: 20px;
|
||||
height: 20px;
|
||||
display: inline-block;
|
||||
border-radius: 12px;
|
||||
vertical-align: sub;
|
||||
transition: .25s;
|
||||
overflow: hidden;
|
||||
.fa {
|
||||
width: 13px!important;
|
||||
}
|
||||
.fa-search {
|
||||
padding-top: 1px;
|
||||
}
|
||||
}
|
||||
|
||||
::v-deep .tree-search .tree-banner-icon {
|
||||
position: absolute;
|
||||
top: 1px;
|
||||
left: 6px;
|
||||
width: 6px;
|
||||
height: 6px;
|
||||
border-radius: 12px;
|
||||
padding: 10px 6px;
|
||||
overflow: hidden;
|
||||
background-color: transparent!important;
|
||||
display: flex;
|
||||
justify-content: center;
|
||||
align-items: center;
|
||||
cursor: pointer;
|
||||
background-color: #D7D8DC;
|
||||
|
||||
.rotate {
|
||||
transition: all .1 .8s;
|
||||
transform: rotate(-90deg);
|
||||
}
|
||||
|
||||
.fa-caret-down {
|
||||
font-size: 16px;
|
||||
}
|
||||
|
||||
.special {
|
||||
top: 1px !important;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
.tree-empty {
|
||||
margin-left: 4px;
|
||||
}
|
||||
|
||||
.fixed-tree-search {
|
||||
margin-bottom: 10px;
|
||||
|
||||
& > > > .el-input__inner {
|
||||
border-radius: 4px;
|
||||
background: #fafafa;
|
||||
padding-right: 32px;
|
||||
}
|
||||
|
||||
::v-deep .tree-search.active {
|
||||
width: 160px;
|
||||
background-color: #ffffff!important;
|
||||
& > > > .el-input__suffix {
|
||||
padding-right: 8px;
|
||||
}
|
||||
|
||||
::v-deep .tree-search.active:hover {
|
||||
border-radius: 12px;
|
||||
& > > > .el-input__prefix {
|
||||
padding-left: 6px;
|
||||
}
|
||||
|
||||
::v-deep .tree-search input {
|
||||
position: relative;
|
||||
left: 20px;
|
||||
width: 133px;
|
||||
height: 100%;
|
||||
background-color: #ffffff!important;
|
||||
& > > > .el-input__suffix-inner {
|
||||
line-height: 30px;
|
||||
}
|
||||
}
|
||||
|
||||
.icon-refresh {
|
||||
border-radius: 4px;
|
||||
padding: 0 1px;
|
||||
z-index: 1;
|
||||
|
||||
&:hover {
|
||||
cursor: pointer;
|
||||
color: #606266;
|
||||
display: flex;
|
||||
justify-content: center;
|
||||
align-items: center;
|
||||
border: none;
|
||||
outline: none;
|
||||
}
|
||||
.tree-header {
|
||||
position: relative;
|
||||
.title {
|
||||
font-weight: 500;
|
||||
}
|
||||
.content {
|
||||
height: 30px;
|
||||
line-height: 30px;
|
||||
border-bottom: 1px solid #e0e0e0;
|
||||
border-radius: 3px;
|
||||
padding: 0 5px;
|
||||
box-sizing: border-box;
|
||||
overflow: hidden;
|
||||
cursor: pointer;
|
||||
background-color: #D7D8DC;
|
||||
.rotate {
|
||||
transition: all .1.8s;
|
||||
transform: rotate(-90deg);
|
||||
}
|
||||
.fa-caret-down {
|
||||
font-size: 16px;
|
||||
}
|
||||
.special {
|
||||
top: 1px!important;
|
||||
}
|
||||
}
|
||||
}
|
||||
.tree-empty {
|
||||
margin-left: 4px;
|
||||
border-color: #d2d2d2;
|
||||
background-color: #e6e6e6;
|
||||
}
|
||||
}
|
||||
|
||||
.icon {
|
||||
cursor: pointer;
|
||||
}
|
||||
|
||||
.tree-action-btn {
|
||||
padding: 0 2px;
|
||||
color: red;
|
||||
}
|
||||
</style>
|
||||
|
||||
@@ -1,4 +1,6 @@
|
||||
<script type="text/jsx">
|
||||
import { toSafeLocalDateStr } from '@/utils/common'
|
||||
|
||||
export default {
|
||||
name: 'ItemValue',
|
||||
props: {
|
||||
@@ -15,27 +17,60 @@ export default {
|
||||
default: null
|
||||
}
|
||||
},
|
||||
computed: {
|
||||
displayValue() {
|
||||
if ([null, undefined, ''].includes(this.value)) {
|
||||
return '-'
|
||||
}
|
||||
if (typeof this.value === 'boolean') {
|
||||
return this.toChoicesDisplay(this.value)
|
||||
} else if (typeof this.value === 'object') {
|
||||
return this.value
|
||||
} else if (this.value instanceof Array) {
|
||||
return this.value.map(item => {
|
||||
if (typeof item === 'object') {
|
||||
return item.label || item.title
|
||||
} else {
|
||||
return item
|
||||
}
|
||||
}).join(', ')
|
||||
} else if (this.isDatetime(this.value)) {
|
||||
return toSafeLocalDateStr(this.value)
|
||||
} else {
|
||||
return this.value
|
||||
}
|
||||
}
|
||||
},
|
||||
methods: {
|
||||
toChoicesDisplay(value) {
|
||||
if (!value) {
|
||||
return this.$t('common.No')
|
||||
}
|
||||
return this.$t('common.Yes')
|
||||
},
|
||||
isDatetime(value) {
|
||||
if (typeof value !== 'string') {
|
||||
return false
|
||||
}
|
||||
if (value.split(' ').length !== 3) {
|
||||
return false
|
||||
}
|
||||
if (value.split(' ')[1].split(':').length !== 3) {
|
||||
return false
|
||||
}
|
||||
if (isNaN(value) && !isNaN(Date.parse(value))) {
|
||||
return true
|
||||
}
|
||||
}
|
||||
},
|
||||
render(h) {
|
||||
if (typeof this.formatter === 'function') {
|
||||
return this.formatter(this.item, this.value)
|
||||
}
|
||||
if (typeof this.value === 'boolean') {
|
||||
return (
|
||||
<span class='item-value'>{this.toChoicesDisplay(this.value)}</span>
|
||||
)
|
||||
}
|
||||
if (this.value instanceof Array) {
|
||||
const newArr = this.value || []
|
||||
return (
|
||||
<span class='item-value'>
|
||||
<span>
|
||||
{
|
||||
newArr.map((item, index) => <div key={index}>{item.key}:{item.value} </div>)
|
||||
}
|
||||
@@ -43,15 +78,14 @@ export default {
|
||||
)
|
||||
}
|
||||
return (
|
||||
<span class='item-value'>{this.value}</span>
|
||||
<span>{this.displayValue}</span>
|
||||
)
|
||||
}
|
||||
}
|
||||
</script>
|
||||
|
||||
<style scoped>
|
||||
.item-value {
|
||||
word-break: break-word;
|
||||
a {
|
||||
color: var(--color-success);
|
||||
}
|
||||
|
||||
</style>
|
||||
|
||||
173
src/components/DetailCard/auto.vue
Normal file
173
src/components/DetailCard/auto.vue
Normal file
@@ -0,0 +1,173 @@
|
||||
<template>
|
||||
<DetailCard v-if="!loading && hasObject && items.length > 0" :items="items" v-bind="$attrs" />
|
||||
</template>
|
||||
|
||||
<script>
|
||||
import DetailCard from './index'
|
||||
import { copy, toSafeLocalDateStr } from '@/utils/common'
|
||||
|
||||
export default {
|
||||
name: 'AutoDetailCard',
|
||||
components: { DetailCard },
|
||||
props: {
|
||||
object: {
|
||||
type: Object,
|
||||
default: () => ({})
|
||||
},
|
||||
url: {
|
||||
type: String,
|
||||
required: true
|
||||
},
|
||||
fields: {
|
||||
type: Array,
|
||||
default: null
|
||||
},
|
||||
excludes: {
|
||||
type: Array,
|
||||
default: null
|
||||
},
|
||||
showUndefine: {
|
||||
type: Boolean,
|
||||
default: true
|
||||
},
|
||||
formatters: {
|
||||
type: Object,
|
||||
default: () => ({})
|
||||
},
|
||||
nested: {
|
||||
type: String,
|
||||
default: null
|
||||
}
|
||||
},
|
||||
data() {
|
||||
return {
|
||||
items: [],
|
||||
loading: true
|
||||
}
|
||||
},
|
||||
computed: {
|
||||
iObject() {
|
||||
if (this.nested) {
|
||||
return this.object[this.nested] || {}
|
||||
} else {
|
||||
return this.object
|
||||
}
|
||||
},
|
||||
hasObject() {
|
||||
return Object.keys(this.iObject).length > 0
|
||||
}
|
||||
},
|
||||
async mounted() {
|
||||
await this.optionAndGenFields()
|
||||
this.loading = false
|
||||
},
|
||||
methods: {
|
||||
defaultFormatter(fields) {
|
||||
const formatter = {}
|
||||
for (const name of fields) {
|
||||
formatter[name] = function(item, val) {
|
||||
if (val === '-') {
|
||||
return <span>{'-'}</span>
|
||||
}
|
||||
return (<span style={{ cursor: 'pointer' }} onClick={() => copy(val)}>
|
||||
{val}
|
||||
</span>)
|
||||
}
|
||||
}
|
||||
return formatter
|
||||
},
|
||||
async optionAndGenFields() {
|
||||
const data = await this.$store.dispatch('common/getUrlMeta', { url: this.url })
|
||||
let remoteMeta = data.actions['GET'] || {}
|
||||
if (this.nested) {
|
||||
remoteMeta = remoteMeta[this.nested]?.children || {}
|
||||
}
|
||||
let fields = this.fields
|
||||
fields = fields || Object.keys(remoteMeta)
|
||||
const defaultExcludes = ['org_id']
|
||||
const excludes = (this.excludes || []).concat(defaultExcludes)
|
||||
fields = fields.filter(item => !excludes.includes(item))
|
||||
const defaultFormatter = this.defaultFormatter(fields)
|
||||
for (const name of fields) {
|
||||
if (typeof name === 'object') {
|
||||
this.items.push(name)
|
||||
continue
|
||||
}
|
||||
const fieldMeta = remoteMeta[name]
|
||||
if (!fieldMeta) {
|
||||
continue
|
||||
}
|
||||
if (fieldMeta['write_only']) {
|
||||
continue
|
||||
}
|
||||
|
||||
let value = this.iObject[name]
|
||||
const label = fieldMeta.label
|
||||
|
||||
if (Array.isArray(value)) {
|
||||
if (typeof value[0] === 'object') {
|
||||
value.forEach(item => {
|
||||
const fieldName = `${name}.${item.name}`
|
||||
if (excludes.includes(fieldName)) {
|
||||
return
|
||||
}
|
||||
this.items.push({
|
||||
key: item.label,
|
||||
value: item.value
|
||||
})
|
||||
})
|
||||
} else if (typeof value[0] === 'string') {
|
||||
value.forEach((item, index) => {
|
||||
let data = {}
|
||||
if (index === 0) {
|
||||
data = {
|
||||
key: label,
|
||||
value: value[index]
|
||||
}
|
||||
} else {
|
||||
data = {
|
||||
value: value[index]
|
||||
}
|
||||
}
|
||||
this.items.push(data)
|
||||
})
|
||||
}
|
||||
continue
|
||||
}
|
||||
if (value === null || value === '') {
|
||||
value = '-'
|
||||
} else if (fieldMeta.type === 'datetime') {
|
||||
value = toSafeLocalDateStr(value)
|
||||
} else if (fieldMeta.type === 'labeled_choice') {
|
||||
value = value?.['label']
|
||||
} else if (fieldMeta.type === 'related_field' || fieldMeta.type === 'nested object') {
|
||||
value = value?.['name']
|
||||
} else if (fieldMeta.type === 'm2m_related_field') {
|
||||
value = value?.map(item => item['name']).join(', ')
|
||||
} else if (fieldMeta.type === 'boolean') {
|
||||
value = value ? this.$t('common.Yes') : this.$t('common.No')
|
||||
}
|
||||
|
||||
if (value === undefined) {
|
||||
if (this.showUndefine) {
|
||||
value = '-'
|
||||
} else {
|
||||
continue
|
||||
}
|
||||
}
|
||||
|
||||
const item = {
|
||||
key: label,
|
||||
value: value,
|
||||
formatter: this.formatters[name] || defaultFormatter[name]
|
||||
}
|
||||
this.items.push(item)
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
</script>
|
||||
|
||||
<style scoped>
|
||||
|
||||
</style>
|
||||
@@ -1,22 +1,11 @@
|
||||
<template>
|
||||
<IBox :title="title" fa="fa-info-circle">
|
||||
<div class="content">
|
||||
<el-row v-if="this.$route.params.id" :gutter="10" class="item">
|
||||
<el-col :span="6"><div :style="{ 'text-align': align }" class="item-label"><label>ID: </label></div></el-col>
|
||||
<el-col :span="18"><div class="item-text">{{ this.$route.params.id }}</div></el-col>
|
||||
</el-row>
|
||||
<el-row v-for="item in items" :key="'card-' + item.key" :gutter="10" class="item">
|
||||
<el-col :span="6">
|
||||
<div :style="{ 'text-align': align }" class="item-label"><label>{{ item.key }}: </label></div>
|
||||
</el-col>
|
||||
<el-col :span="18">
|
||||
<div class="item-text">
|
||||
<ItemValue :value="item.value" v-bind="item" />
|
||||
</div>
|
||||
</el-col>
|
||||
</el-row>
|
||||
<slot />
|
||||
</div>
|
||||
<IBox :title="title" :fa="fa">
|
||||
<el-form class="content" label-position="left" label-width="25%">
|
||||
<el-form-item v-for="item in items" :key="item.key" :label="item.key">
|
||||
<ItemValue class="item-value" :value="item.value" v-bind="item" />
|
||||
</el-form-item>
|
||||
</el-form>
|
||||
<slot />
|
||||
</IBox>
|
||||
</template>
|
||||
|
||||
@@ -34,6 +23,10 @@ export default {
|
||||
return this.$t('common.BasicInfo')
|
||||
}
|
||||
},
|
||||
fa: {
|
||||
type: String,
|
||||
default: 'fa-info-circle'
|
||||
},
|
||||
items: {
|
||||
type: Array,
|
||||
default: () => []
|
||||
@@ -46,7 +39,42 @@ export default {
|
||||
}
|
||||
</script>
|
||||
|
||||
<style scoped>
|
||||
<style lang="scss" scoped>
|
||||
.el-card__body {
|
||||
padding: 20px 40px;
|
||||
}
|
||||
|
||||
.el-form-item {
|
||||
border-bottom: 1px dashed #EBEEF5;
|
||||
padding: 1px 0;
|
||||
margin-bottom: 0;
|
||||
|
||||
&:last-child {
|
||||
border-bottom: none;
|
||||
}
|
||||
|
||||
&:hover {
|
||||
}
|
||||
|
||||
>>> .el-form-item__label {
|
||||
padding-right: 8%;
|
||||
white-space: nowrap;
|
||||
text-overflow: ellipsis;
|
||||
overflow: hidden;
|
||||
}
|
||||
|
||||
>>> .el-form-item__content {
|
||||
font-size: 13px;
|
||||
}
|
||||
|
||||
>>> .el-tag--mini {
|
||||
margin-right: 3px;
|
||||
}
|
||||
}
|
||||
|
||||
.item-value span {
|
||||
word-break: break-word;
|
||||
}
|
||||
.content {
|
||||
font-size: 13px;
|
||||
line-height: 2.5;
|
||||
|
||||
105
src/components/Dialog/DiffDetail.vue
Normal file
105
src/components/Dialog/DiffDetail.vue
Normal file
@@ -0,0 +1,105 @@
|
||||
<template>
|
||||
<Dialog
|
||||
v-if="detailVisible"
|
||||
:show-cancel="false"
|
||||
:show-confirm="false"
|
||||
:title="title"
|
||||
:visible.sync="detailVisible"
|
||||
>
|
||||
<div>
|
||||
<div v-if="isEmpty()" style="text-align: center">
|
||||
{{ this.$tc('common.NoContent') }}
|
||||
</div>
|
||||
<div v-else>
|
||||
<el-table
|
||||
:data="diff"
|
||||
class="diffTable"
|
||||
>
|
||||
<el-table-column
|
||||
:label="$tc('audits.ChangeField')"
|
||||
:prop="fieldName"
|
||||
show-overflow-tooltip
|
||||
width="100"
|
||||
/>
|
||||
<el-table-column
|
||||
:label="$tc('audits.BeforeChange')"
|
||||
:prop="leftKeyName"
|
||||
show-overflow-tooltip
|
||||
/>
|
||||
<el-table-column
|
||||
:label="$tc('audits.AfterChange')"
|
||||
:prop="rightKeyName"
|
||||
show-overflow-tooltip
|
||||
/>
|
||||
</el-table>
|
||||
</div>
|
||||
</div>
|
||||
</Dialog>
|
||||
</template>
|
||||
|
||||
<script>
|
||||
import Dialog from '@/components/Dialog/index'
|
||||
|
||||
export default {
|
||||
name: 'DiffDetail',
|
||||
components: {
|
||||
Dialog
|
||||
},
|
||||
props: {
|
||||
title: {
|
||||
type: String,
|
||||
default: () => ''
|
||||
},
|
||||
fieldName: {
|
||||
type: String,
|
||||
default: () => 'field'
|
||||
},
|
||||
leftKeyName: {
|
||||
type: String,
|
||||
default: () => 'before'
|
||||
},
|
||||
rightKeyName: {
|
||||
type: String,
|
||||
default: () => 'after'
|
||||
}
|
||||
},
|
||||
data() {
|
||||
return {
|
||||
diff: [],
|
||||
detailVisible: false
|
||||
}
|
||||
},
|
||||
methods: {
|
||||
isEmpty() {
|
||||
const content = this.diff
|
||||
return !content || JSON.stringify(content) === '{}'
|
||||
},
|
||||
show(data) {
|
||||
this.diff = data
|
||||
this.detailVisible = true
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
</script>
|
||||
|
||||
<style lang='scss' scoped>
|
||||
.el-tag {
|
||||
width: 100%;
|
||||
white-space: normal;
|
||||
height: auto;
|
||||
}
|
||||
|
||||
.el-table::before {
|
||||
background-color: inherit;
|
||||
}
|
||||
|
||||
.diffTable {
|
||||
width: 100%;
|
||||
max-height: 80vh;
|
||||
|
||||
& >>> td {
|
||||
padding: 5px 0 !important;
|
||||
}
|
||||
}
|
||||
</style>
|
||||
@@ -1,19 +1,21 @@
|
||||
<template>
|
||||
<el-dialog
|
||||
:append-to-body="true"
|
||||
:modal-append-to-body="true"
|
||||
:title="title"
|
||||
:top="top"
|
||||
:width="iWidth"
|
||||
class="dialog"
|
||||
v-bind="$attrs"
|
||||
:append-to-body="false"
|
||||
:modal-append-to-body="false"
|
||||
v-on="$listeners"
|
||||
>
|
||||
<slot />
|
||||
<div slot="footer" class="dialog-footer">
|
||||
<slot name="footer">
|
||||
<el-button v-if="showCancel" size="small" @click="onCancel">{{ cancelTitle }}</el-button>
|
||||
<el-button v-if="showConfirm" type="primary" size="small" :loading="loadingStatus" @click="onConfirm">{{ confirmTitle }}</el-button>
|
||||
<el-button v-if="showCancel && showButtons" @click="onCancel">{{ cancelTitle }}</el-button>
|
||||
<el-button v-if="showConfirm && showButtons" :loading="loadingStatus" type="primary" @click="onConfirm">
|
||||
{{ confirmTitle }}
|
||||
</el-button>
|
||||
</slot>
|
||||
</div>
|
||||
</el-dialog>
|
||||
@@ -27,16 +29,6 @@ export default {
|
||||
type: String,
|
||||
default: 'Title'
|
||||
},
|
||||
showCancel: {
|
||||
type: Boolean,
|
||||
default: true
|
||||
},
|
||||
cancelTitle: {
|
||||
type: String,
|
||||
default() {
|
||||
return this.$t('common.Cancel')
|
||||
}
|
||||
},
|
||||
top: {
|
||||
type: String,
|
||||
default: '3vh'
|
||||
@@ -49,20 +41,37 @@ export default {
|
||||
type: Boolean,
|
||||
default: true
|
||||
},
|
||||
loadingStatus: {
|
||||
type: Boolean,
|
||||
default: false
|
||||
},
|
||||
confirmTitle: {
|
||||
type: String,
|
||||
default() {
|
||||
return this.$t('common.Confirm')
|
||||
}
|
||||
},
|
||||
showCancel: {
|
||||
type: Boolean,
|
||||
default: true
|
||||
},
|
||||
cancelTitle: {
|
||||
type: String,
|
||||
default() {
|
||||
return this.$t('common.Cancel')
|
||||
}
|
||||
},
|
||||
showButtons: {
|
||||
type: Boolean,
|
||||
default: true
|
||||
},
|
||||
loadingStatus: {
|
||||
type: Boolean,
|
||||
default: false
|
||||
},
|
||||
maxWidth: {
|
||||
type: String,
|
||||
default: '1200px'
|
||||
}
|
||||
},
|
||||
data() {
|
||||
return {
|
||||
}
|
||||
return {}
|
||||
},
|
||||
computed: {
|
||||
iWidth() {
|
||||
@@ -81,12 +90,38 @@ export default {
|
||||
</script>
|
||||
|
||||
<style lang="scss" scoped>
|
||||
.dialog >>> .el-dialog__header {
|
||||
/*padding-top: 10px;*/
|
||||
.dialog >>> .el-dialog {
|
||||
border-radius: 0.3em;
|
||||
max-width: 1500px;
|
||||
|
||||
.el-icon-circle-check {
|
||||
display: none;
|
||||
}
|
||||
|
||||
&__header {
|
||||
box-sizing: border-box;
|
||||
padding: 15px 22px;
|
||||
border-bottom: 1px solid #dee2e6;
|
||||
font-weight: 400;
|
||||
}
|
||||
|
||||
&__body {
|
||||
padding: 20px 30px;
|
||||
|
||||
&:has(.el-table) {
|
||||
background: #f3f3f4;
|
||||
}
|
||||
}
|
||||
|
||||
&__footer {
|
||||
border-top: 1px solid #dee2e6;
|
||||
padding: 16px;
|
||||
justify-content: flex-end;
|
||||
}
|
||||
}
|
||||
|
||||
.dialog-footer {
|
||||
padding-right: 20px;
|
||||
.dialog-footer >>> button.el-button {
|
||||
font-size: 13px;
|
||||
padding: 10px 20px;
|
||||
}
|
||||
|
||||
</style>
|
||||
|
||||
109
src/components/FormFields/BasicTree.vue
Normal file
109
src/components/FormFields/BasicTree.vue
Normal file
@@ -0,0 +1,109 @@
|
||||
<template>
|
||||
<el-tree
|
||||
:data="iTree"
|
||||
:default-checked-keys="iValue"
|
||||
:default-expand-all="true"
|
||||
:default-expanded-keys="iValue"
|
||||
:props="defaultProps"
|
||||
:render-content="renderContent"
|
||||
class="el-tree-custom"
|
||||
node-key="value"
|
||||
show-checkbox
|
||||
@check="handleCheckChange"
|
||||
/>
|
||||
</template>
|
||||
|
||||
<script>
|
||||
export default {
|
||||
props: {
|
||||
value: {
|
||||
type: Array,
|
||||
default: () => []
|
||||
},
|
||||
tree: {
|
||||
type: Array,
|
||||
default: () => []
|
||||
},
|
||||
readonly: {
|
||||
type: Boolean,
|
||||
default: false
|
||||
}
|
||||
},
|
||||
data() {
|
||||
return {
|
||||
defaultProps: {
|
||||
children: 'children',
|
||||
label: 'label'
|
||||
}
|
||||
}
|
||||
},
|
||||
computed: {
|
||||
iValue() {
|
||||
return this.value.map(item => {
|
||||
if (item.value) {
|
||||
return item.value
|
||||
}
|
||||
return item
|
||||
})
|
||||
},
|
||||
iTree() {
|
||||
if (!this.readonly) {
|
||||
return this.tree
|
||||
} else {
|
||||
return this.setTreeReadonly(this.tree)
|
||||
}
|
||||
}
|
||||
},
|
||||
methods: {
|
||||
handleCheckChange(node, { checkedNodes }) {
|
||||
const checkedKeys = checkedNodes
|
||||
.filter(item => !item.children)
|
||||
.map(node => node.value)
|
||||
this.$emit('input', checkedKeys)
|
||||
},
|
||||
setTreeReadonly(tree) {
|
||||
return tree.map(item => {
|
||||
item.disabled = true
|
||||
if (item.children) {
|
||||
item.children = this.setTreeReadonly(item.children)
|
||||
}
|
||||
return item
|
||||
})
|
||||
},
|
||||
renderContent(h, { node, data, store }) {
|
||||
let label = node.label
|
||||
let helpText = ''
|
||||
const regex = /(.*?)\s*\((.*?)\)/
|
||||
const match = label.match(regex)
|
||||
if (match) {
|
||||
label = match[1]
|
||||
helpText = match[2]
|
||||
}
|
||||
|
||||
return (
|
||||
<span >
|
||||
<span>{label} </span>
|
||||
{helpText
|
||||
? (<el-tooltip content={helpText} placement='top'>
|
||||
<i class='fa fa-info-circle'></i>
|
||||
</el-tooltip>) : ''}
|
||||
</span>)
|
||||
}
|
||||
}
|
||||
}
|
||||
</script>
|
||||
|
||||
<style lang="scss" scoped>
|
||||
|
||||
.el-tree-custom >>> {
|
||||
.help-tips {
|
||||
margin-left: 10px;
|
||||
font-size: 12px;
|
||||
color: #999;
|
||||
}
|
||||
.el-tree-node__content:hover {
|
||||
background-color: inherit;
|
||||
}
|
||||
}
|
||||
|
||||
</style>
|
||||
283
src/components/FormFields/CodeEditor.vue
Normal file
283
src/components/FormFields/CodeEditor.vue
Normal file
@@ -0,0 +1,283 @@
|
||||
<template>
|
||||
<div class="code-editor" style="font-size: 12px">
|
||||
<div class="toolbar">
|
||||
<div
|
||||
v-for="(item,index) in toolbar.left"
|
||||
:key="index"
|
||||
style="display: inline-block; margin: 0 2px"
|
||||
>
|
||||
<el-tooltip :content="item.tip" :disabled="!item.tip" placement="top">
|
||||
<el-button
|
||||
v-if="item.type ==='button'"
|
||||
:disabled="item.disabled"
|
||||
:type="item.el&&item.el.type"
|
||||
size="mini"
|
||||
@click="item.callback()"
|
||||
>
|
||||
<i :class="item.icon" style="margin-right: 4px;" />{{ item.name }}
|
||||
</el-button>
|
||||
|
||||
<el-autocomplete
|
||||
v-if="item.type === 'input' && item.el.autoComplete"
|
||||
v-model="item.value"
|
||||
:placeholder="item.placeholder"
|
||||
:fetch-suggestions="item.el.query"
|
||||
class="inline-input"
|
||||
size="mini"
|
||||
@select="item.callback(item.value)"
|
||||
@change="item.callback(item.value)"
|
||||
/>
|
||||
|
||||
<div v-if="item.type==='select' && item.el && item.el.create" class="select-content">
|
||||
<span class="filter-label">
|
||||
{{ item.name }}:
|
||||
</span>
|
||||
<el-select
|
||||
v-if="item.type==='select' && item.el && item.el.create"
|
||||
:key="index"
|
||||
v-model="item.value"
|
||||
:allow-create="item.el.create || false"
|
||||
:filterable="item.el.create || false"
|
||||
:multiple="item.el.multiple"
|
||||
:placeholder="item.name"
|
||||
class="autoWidth-select"
|
||||
default-first-option
|
||||
size="mini"
|
||||
@change="item.callback(item.value)"
|
||||
>
|
||||
<template slot="prefix">
|
||||
{{ item.label + ':' + item.value }}
|
||||
</template>
|
||||
<el-option
|
||||
v-for="(option,id) in item.options"
|
||||
:key="id"
|
||||
:label="option.label"
|
||||
:title="option.value"
|
||||
:value="option.value"
|
||||
/>
|
||||
</el-select>
|
||||
</div>
|
||||
<el-dropdown
|
||||
v-if="item.type==='select' && (!item.el || !item.el.create) "
|
||||
trigger="click"
|
||||
@command="(command) => {
|
||||
item.value= command
|
||||
item.callback(command)
|
||||
}"
|
||||
>
|
||||
<el-button size="mini" type="default">
|
||||
<b>{{ item.name }}:</b> {{ getLabel(item.value, item.options) }} <i
|
||||
class="el-icon-arrow-down el-icon--right"
|
||||
/>
|
||||
</el-button>
|
||||
<el-dropdown-menu v-slot="dropdown">
|
||||
<el-dropdown-item v-for="(option,i) in item.options" :key="i" :command="option.value">
|
||||
{{ option.label }}
|
||||
</el-dropdown-item>
|
||||
</el-dropdown-menu>
|
||||
</el-dropdown>
|
||||
|
||||
<el-switch
|
||||
v-if="item.type === 'switch'"
|
||||
v-model="item.value"
|
||||
:active-text="item.name"
|
||||
:disabled="item.disabled"
|
||||
@change="item.callback( item.value)"
|
||||
/>
|
||||
</el-tooltip>
|
||||
</div>
|
||||
|
||||
<div class="right-side" style="float: right">
|
||||
<div
|
||||
v-for="(item,index) in toolbar.right"
|
||||
:key="index"
|
||||
style="display: inline-block"
|
||||
>
|
||||
<el-tooltip :content="item.tip">
|
||||
<el-button
|
||||
v-if="item.type ==='button'"
|
||||
:disabled="item.disabled"
|
||||
size="mini"
|
||||
style="background-color: transparent"
|
||||
type="default"
|
||||
@click="item.callback()"
|
||||
>
|
||||
<i v-if="item.icon.startsWith('fa')" :class="'fa ' + item.icon" />
|
||||
<svg-icon v-else :icon-class="item.icon" style="font-size: 14px;" />
|
||||
</el-button>
|
||||
</el-tooltip>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
<codemirror ref="myCm" v-model="iValue" :options="iOptions" class="editor" />
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<script>
|
||||
import { codemirror } from 'vue-codemirror'
|
||||
|
||||
import 'codemirror/mode/shell/shell'
|
||||
import 'codemirror/mode/powershell/powershell'
|
||||
import 'codemirror/mode/python/python'
|
||||
import 'codemirror/mode/yaml/yaml'
|
||||
import 'codemirror/mode/ruby/ruby' // theme css
|
||||
import 'codemirror/theme/base16-light.css'
|
||||
import 'codemirror/theme/idea.css'
|
||||
import 'codemirror/theme/mbo.css'
|
||||
import 'codemirror/theme/duotone-light.css'
|
||||
import 'codemirror/lib/codemirror.css'
|
||||
|
||||
export default {
|
||||
components: {
|
||||
codemirror
|
||||
},
|
||||
props: {
|
||||
toolbar: {
|
||||
type: [Array, Object],
|
||||
default: () => []
|
||||
},
|
||||
value: {
|
||||
type: [String, Object],
|
||||
default: () => ''
|
||||
},
|
||||
options: {
|
||||
type: Object,
|
||||
default: () => {
|
||||
return {}
|
||||
}
|
||||
}
|
||||
},
|
||||
data() {
|
||||
return {}
|
||||
},
|
||||
computed: {
|
||||
iValue: {
|
||||
get() {
|
||||
return this.value
|
||||
},
|
||||
set(val) {
|
||||
this.$emit('update:value', val)
|
||||
this.$emit('change', val)
|
||||
}
|
||||
},
|
||||
iOptions() {
|
||||
const defaultOptions = {
|
||||
tabSize: 4,
|
||||
mode: 'shell',
|
||||
lineNumbers: true,
|
||||
theme: 'idea',
|
||||
placeholder: 'Code goes here...',
|
||||
autofocus: true
|
||||
}
|
||||
return Object.assign(defaultOptions, this.options)
|
||||
}
|
||||
},
|
||||
methods: {
|
||||
getLabel(value, items) {
|
||||
for (const item of items) {
|
||||
if (item.value === value) {
|
||||
return item.label
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
</script>
|
||||
|
||||
<style lang="scss" scoped>
|
||||
.editor {
|
||||
border: solid 1px #f3f3f3;
|
||||
}
|
||||
|
||||
.toolbar {
|
||||
height: 100%;
|
||||
width: 100%;
|
||||
line-height: 29px;
|
||||
vertical-align: bottom;
|
||||
display: inline-block;
|
||||
padding: 3px 3px 3px 0;
|
||||
margin-bottom: 5px;
|
||||
}
|
||||
|
||||
> > > .CodeMirror pre.CodeMirror-line,
|
||||
> > > .CodeMirror-linenumber.CodeMirror-gutter-elt {
|
||||
line-height: 18px !important;
|
||||
}
|
||||
|
||||
.runas-input {
|
||||
height: 28px;
|
||||
|
||||
> > > {
|
||||
.el-select {
|
||||
width: 100px;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
.right-side {
|
||||
.el-button {
|
||||
border: none;
|
||||
padding: 2px;
|
||||
font-size: 14px;
|
||||
width: 26px;
|
||||
height: 26px;
|
||||
color: #888;
|
||||
background-color: transparent;
|
||||
margin-left: 2px;
|
||||
}
|
||||
}
|
||||
|
||||
.autoWidth-select {
|
||||
min-width: 100px;
|
||||
}
|
||||
|
||||
.autoWidth-select > > > .el-input__prefix {
|
||||
position: relative;
|
||||
left: 0px;
|
||||
box-sizing: border-box;
|
||||
height: 28px;
|
||||
line-height: 28px;
|
||||
visibility: hidden;
|
||||
}
|
||||
|
||||
.autoWidth-select > > > input {
|
||||
position: absolute;
|
||||
padding-left: 0px;
|
||||
border: none;
|
||||
color: #606266;
|
||||
background-color: #e6e6e6;
|
||||
font-size: 12px;
|
||||
font-weight: 470;
|
||||
line-height: 27px;
|
||||
}
|
||||
|
||||
> > > .el-select {
|
||||
top: -1px;
|
||||
|
||||
.el-input .el-select__caret {
|
||||
color: #7a7c7f;
|
||||
}
|
||||
}
|
||||
|
||||
> > > .el-button.el-button--default {
|
||||
background-color: #e6e6e6;
|
||||
}
|
||||
|
||||
.filter-label {
|
||||
font-size: 12px;
|
||||
font-weight: 700;
|
||||
}
|
||||
|
||||
.select-content {
|
||||
display: inline-block;
|
||||
position: relative;
|
||||
top: 1px;
|
||||
height: 28px;
|
||||
line-height: 28px;
|
||||
padding-left: 15px;
|
||||
font-size: 0;
|
||||
border: 1px solid #DCDFE6;
|
||||
border-radius: 4px;
|
||||
background-color: #e6e6e6;
|
||||
}
|
||||
</style>
|
||||
@@ -2,9 +2,9 @@
|
||||
<el-date-picker
|
||||
v-model="value"
|
||||
type="datetimerange"
|
||||
:range-separator="this.$t('common.To')"
|
||||
:start-placeholder="this.$t('common.DateStart')"
|
||||
:end-placeholder="this.$t('common.DateEnd')"
|
||||
:range-separator="$tc('common.To')"
|
||||
:start-placeholder="$tc('common.DateStart')"
|
||||
:end-placeholder="$tc('common.DateEnd')"
|
||||
size="small"
|
||||
:clearable="false"
|
||||
class="datepicker"
|
||||
@@ -41,37 +41,23 @@ export default {
|
||||
shortcuts: [
|
||||
{
|
||||
text: this.$t('common.DateLast24Hours'),
|
||||
onClick(picker) {
|
||||
const end = new Date()
|
||||
const start = new Date()
|
||||
start.setTime(start.getTime() - 3600 * 1000 * 24)
|
||||
picker.$emit('pick', [start, end])
|
||||
}
|
||||
onClick: (picker) => this.onShortcutClick(picker, 1)
|
||||
},
|
||||
{
|
||||
text: this.$t('common.DateLastWeek'),
|
||||
onClick(picker) {
|
||||
const end = new Date()
|
||||
const start = new Date()
|
||||
start.setTime(start.getTime() - 3600 * 1000 * 24 * 7)
|
||||
picker.$emit('pick', [start, end])
|
||||
}
|
||||
onClick: (picker) => this.onShortcutClick(picker, 7)
|
||||
}, {
|
||||
text: this.$t('common.DateLastMonth'),
|
||||
onClick(picker) {
|
||||
const end = new Date()
|
||||
const start = new Date()
|
||||
start.setTime(start.getTime() - 3600 * 1000 * 24 * 30)
|
||||
picker.$emit('pick', [start, end])
|
||||
}
|
||||
onClick: (picker) => this.onShortcutClick(picker, 30)
|
||||
}, {
|
||||
text: this.$t('common.DateLast3Months'),
|
||||
onClick(picker) {
|
||||
const end = new Date()
|
||||
const start = new Date()
|
||||
start.setTime(start.getTime() - 3600 * 1000 * 24 * 90)
|
||||
picker.$emit('pick', [start, end])
|
||||
}
|
||||
onClick: (picker) => this.onShortcutClick(picker, 90)
|
||||
}, {
|
||||
text: this.$t('common.DateLastHarfYear'),
|
||||
onClick: (picker) => this.onShortcutClick(picker, 183)
|
||||
}, {
|
||||
text: this.$t('common.DateLastYear'),
|
||||
onClick: (picker) => this.onShortcutClick(picker, 365)
|
||||
}
|
||||
]
|
||||
}
|
||||
@@ -86,28 +72,38 @@ export default {
|
||||
this.$log.debug('Date change: ', val)
|
||||
this.$emit('dateChange', val)
|
||||
}
|
||||
},
|
||||
onShortcutClick(picker, day) {
|
||||
const end = new Date()
|
||||
const start = new Date()
|
||||
start.setTime(start.getTime() - 3600 * 1000 * 24 * day)
|
||||
picker.$emit('pick', [start, end])
|
||||
}
|
||||
}
|
||||
}
|
||||
</script>
|
||||
|
||||
<style lang='scss' scoped>
|
||||
.datepicker{
|
||||
.datepicker {
|
||||
width: 233px;
|
||||
&>>> .el-range__icon {
|
||||
|
||||
& >>> .el-range__icon {
|
||||
margin-top: 2px;
|
||||
margin-right: 3px;
|
||||
}
|
||||
&>>> .el-range-input {
|
||||
|
||||
& >>> .el-range-input {
|
||||
width: 49%;
|
||||
}
|
||||
}
|
||||
.el-input__inner{
|
||||
|
||||
.el-input__inner {
|
||||
border: 1px solid #dcdee2;
|
||||
border-radius: 3px;
|
||||
height: 32x;
|
||||
height: 32px;
|
||||
}
|
||||
.el-date-editor ::v-deep .el-range-separator{
|
||||
|
||||
.el-date-editor ::v-deep .el-range-separator {
|
||||
line-height: 28px;
|
||||
}
|
||||
</style>
|
||||
|
||||
83
src/components/FormFields/DynamicInput.vue
Normal file
83
src/components/FormFields/DynamicInput.vue
Normal file
@@ -0,0 +1,83 @@
|
||||
<template>
|
||||
<div>
|
||||
<div v-for="(command, index) in value" :key="index" :prop="'value.' + index + '.value'" class="command-item">
|
||||
<el-input v-model="value[index]" size="mini">
|
||||
<template slot="prepend"> {{ inputTitle + ' ' + (index + 1) }}</template>
|
||||
</el-input>
|
||||
<div class="input-button">
|
||||
<el-button
|
||||
:disabled="deleteDisabled()"
|
||||
icon="el-icon-minus"
|
||||
size="mini"
|
||||
style="flex-shrink: 0;"
|
||||
type="danger"
|
||||
@click="handleDelete(command)"
|
||||
/>
|
||||
<el-button
|
||||
v-if="index === value.length - 1"
|
||||
icon="el-icon-plus"
|
||||
size="mini"
|
||||
style="flex-shrink: 0;"
|
||||
type="primary"
|
||||
@click="handleAdd()"
|
||||
/>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<script>
|
||||
export default {
|
||||
props: {
|
||||
value: {
|
||||
type: Array,
|
||||
default: () => ['']
|
||||
},
|
||||
inputTitle: {
|
||||
type: String,
|
||||
default: () => ''
|
||||
}
|
||||
},
|
||||
data() {
|
||||
return {}
|
||||
},
|
||||
methods: {
|
||||
handleDelete(command) {
|
||||
const index = this.value.indexOf(command)
|
||||
if (index !== -1) {
|
||||
this.value.splice(index, 1)
|
||||
}
|
||||
},
|
||||
handleAdd() {
|
||||
this.value.push('')
|
||||
},
|
||||
deleteDisabled() {
|
||||
return this.value.length <= 1
|
||||
}
|
||||
}
|
||||
}
|
||||
</script>
|
||||
|
||||
<style lang="scss" scoped>
|
||||
.el-input {
|
||||
width: 85%;
|
||||
}
|
||||
.command-item {
|
||||
display: flex;
|
||||
margin: 5px 0;
|
||||
}
|
||||
.input-button {
|
||||
margin-top: 2px;
|
||||
display: flex;
|
||||
margin-left: 20px
|
||||
}
|
||||
.input-button ::v-deep .el-button.el-button--mini {
|
||||
height: 25px;
|
||||
padding: 5px;
|
||||
}
|
||||
.el-input-group__append .el-button {
|
||||
font-size: 14px;
|
||||
color: #1a1a1a;
|
||||
padding: 9px 20px;
|
||||
}
|
||||
</style>
|
||||
@@ -0,0 +1,161 @@
|
||||
<template>
|
||||
<Dialog
|
||||
:destroy-on-close="true"
|
||||
:show-buttons="false"
|
||||
:title="$tc('common.SelectAttrs')"
|
||||
v-bind="$attrs"
|
||||
v-on="$listeners"
|
||||
>
|
||||
<div v-if="!loading">
|
||||
<DataForm
|
||||
:form="form"
|
||||
class="attr-form"
|
||||
v-bind="formConfig"
|
||||
@submit="onAttrDialogConfirm"
|
||||
/>
|
||||
</div>
|
||||
</Dialog>
|
||||
</template>
|
||||
|
||||
<script>
|
||||
import DataForm from '@/components/DataForm/index.vue'
|
||||
import Dialog from '@/components/Dialog/index.vue'
|
||||
import ValueField from '@/components/FormFields/JSONManyToManySelect/ValueField.vue'
|
||||
import { attrMatchOptions, typeMatchMapper } from './const'
|
||||
|
||||
export default {
|
||||
name: 'AttrFormDialog',
|
||||
components: { Dialog, DataForm },
|
||||
props: {
|
||||
attrs: {
|
||||
type: Array,
|
||||
default: () => ([])
|
||||
},
|
||||
attrsAdded: {
|
||||
type: Array,
|
||||
default: () => ([])
|
||||
},
|
||||
form: {
|
||||
type: Object,
|
||||
default: () => ({})
|
||||
}
|
||||
},
|
||||
data() {
|
||||
const vm = this
|
||||
return {
|
||||
loading: true,
|
||||
currentValue: '',
|
||||
formConfig: {
|
||||
// 为了方便更新,避免去取 fields 的索引
|
||||
hasSaveContinue: false,
|
||||
fields: [
|
||||
{
|
||||
id: 'name',
|
||||
label: this.$t('common.AttrName'),
|
||||
type: 'select',
|
||||
options: this.attrs.map(attr => {
|
||||
let disabled = this.attrsAdded.includes(attr.name) && this.form.name !== attr.name
|
||||
if (attr.disabled) {
|
||||
disabled = true
|
||||
}
|
||||
return { label: attr.label, value: attr.name, disabled: disabled }
|
||||
}),
|
||||
on: {
|
||||
change: ([val], updateForm) => {
|
||||
// 变化会影响 match 的选项
|
||||
const attr = this.attrs.find(attr => attr.name === val)
|
||||
if (!attr) return
|
||||
const matchOption = vm.updateMatchOptions(attr)
|
||||
setTimeout(() => {
|
||||
updateForm({ match: matchOption.value })
|
||||
}, 10)
|
||||
}
|
||||
}
|
||||
},
|
||||
{
|
||||
id: 'match',
|
||||
label: this.$t('common.Match'),
|
||||
type: 'select',
|
||||
options: attrMatchOptions,
|
||||
on: {
|
||||
change: ([value], updateForm) => {
|
||||
// 变化会影响 value 的选项
|
||||
setTimeout(() => {
|
||||
this.formConfig.fields[2].el.match = value
|
||||
}, 10)
|
||||
}
|
||||
}
|
||||
},
|
||||
{
|
||||
id: 'value',
|
||||
label: this.$t('common.AttrValue'),
|
||||
component: ValueField,
|
||||
el: {
|
||||
match: attrMatchOptions[0].value,
|
||||
attr: this.attrs[0]
|
||||
},
|
||||
on: {
|
||||
input: ([value], updateForm) => {
|
||||
vm.currentValue = value
|
||||
}
|
||||
}
|
||||
}
|
||||
]
|
||||
}
|
||||
}
|
||||
},
|
||||
mounted() {
|
||||
if (this.form.index === undefined || this.form.index === -1) {
|
||||
Object.assign(this.form, this.getDefaultAttrForm())
|
||||
}
|
||||
this.updateMatchOptions()
|
||||
this.$log.debug('Attr Form config: ', this.formConfig)
|
||||
this.loading = false
|
||||
},
|
||||
methods: {
|
||||
getDefaultAttrForm() {
|
||||
const attrKeys = this.attrs.map(attr => attr.name)
|
||||
const diff = attrKeys.filter(attr => !this.attrsAdded.includes(attr))
|
||||
let name = this.attrs[0].name
|
||||
if (diff.length > 0) {
|
||||
name = diff[0]
|
||||
}
|
||||
return {
|
||||
name: name,
|
||||
match: 'exact',
|
||||
value: '',
|
||||
rel: 'and'
|
||||
}
|
||||
},
|
||||
onAttrDialogConfirm(form) {
|
||||
this.$emit('confirm', form)
|
||||
},
|
||||
updateMatchOptions(attr) {
|
||||
if (!attr) {
|
||||
attr = this.attrs.find(attr => attr.name === this.form.name)
|
||||
}
|
||||
if (!attr) return
|
||||
const attrType = attr.type || 'str'
|
||||
const matchSupports = typeMatchMapper[attrType]
|
||||
attrMatchOptions.forEach((option) => {
|
||||
option.hidden = !matchSupports.includes(option.value)
|
||||
})
|
||||
this.formConfig.fields[2].el.attr = attr
|
||||
const supports = attrMatchOptions.filter(option => !option.hidden)
|
||||
const matchOption = supports.find(item => item.value === this.form.match) || supports[0]
|
||||
this.formConfig.fields[2].el.match = matchOption.value
|
||||
return matchOption
|
||||
}
|
||||
}
|
||||
}
|
||||
</script>
|
||||
|
||||
<style lang="scss" scoped>
|
||||
|
||||
.attr-form {
|
||||
>>> .el-select {
|
||||
width: 100%;
|
||||
}
|
||||
}
|
||||
</style>
|
||||
|
||||
@@ -0,0 +1,77 @@
|
||||
<template>
|
||||
<Dialog
|
||||
:destroy-on-close="true"
|
||||
:show-buttons="false"
|
||||
:title="$tc('common.MatchResult')"
|
||||
:v-bind="$attrs"
|
||||
:v-on="$listeners"
|
||||
:visible.sync="iVisible"
|
||||
>
|
||||
<ListTable v-bind="attrMatchTableConfig" />
|
||||
</Dialog>
|
||||
</template>
|
||||
|
||||
<script>
|
||||
import Dialog from '@/components/Dialog/index.vue'
|
||||
import ListTable from '@/components/ListTable/index.vue'
|
||||
|
||||
export default {
|
||||
name: 'AttrMatchResultDialog',
|
||||
components: { ListTable, Dialog },
|
||||
props: {
|
||||
url: {
|
||||
type: String,
|
||||
default: ''
|
||||
},
|
||||
attrs: {
|
||||
type: Array,
|
||||
default: () => ([])
|
||||
},
|
||||
visible: {
|
||||
type: Boolean,
|
||||
default: false
|
||||
}
|
||||
},
|
||||
data() {
|
||||
return {
|
||||
attrMatchTableConfig: {
|
||||
headerActions: {
|
||||
hasCreate: false,
|
||||
hasImport: false,
|
||||
hasExport: false,
|
||||
hasMoreActions: false
|
||||
},
|
||||
tableConfig: {
|
||||
url: this.url,
|
||||
columns: this.attrs.filter(item => item.inTable).map(item => {
|
||||
return {
|
||||
prop: item.name,
|
||||
label: item.label,
|
||||
formatter: item.formatter
|
||||
}
|
||||
}),
|
||||
columnsMeta: {
|
||||
actions: {
|
||||
has: false
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
},
|
||||
computed: {
|
||||
iVisible: {
|
||||
set(val) {
|
||||
this.$emit('update:visible', val)
|
||||
},
|
||||
get() {
|
||||
return this.visible
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
</script>
|
||||
|
||||
<style scoped>
|
||||
|
||||
</style>
|
||||
110
src/components/FormFields/JSONManyToManySelect/ValueField.vue
Normal file
110
src/components/FormFields/JSONManyToManySelect/ValueField.vue
Normal file
@@ -0,0 +1,110 @@
|
||||
<template>
|
||||
<div v-if="!loading">
|
||||
<TagInput v-if="type === 'array'" :value="iValue" @input="handleInput" />
|
||||
<Select2 v-else-if="type === 'select'" :value="iValue" v-bind="attr.el" @change="handleInput" @input="handleInput" />
|
||||
<Switcher v-else-if="type === 'bool'" :value="iValue" @change="handleInput" @input="handleInput" />
|
||||
<el-input v-else :value="iValue" @input="handleInput" />
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<script>
|
||||
import TagInput from '@/components/FormFields/TagInput.vue'
|
||||
import Select2 from '@/components/FormFields/Select2.vue'
|
||||
import Switcher from '@/components/FormFields/Switcher.vue'
|
||||
|
||||
export default {
|
||||
name: 'ValueField',
|
||||
components: { Switcher, TagInput, Select2 },
|
||||
props: {
|
||||
value: {
|
||||
type: [String, Number, Boolean, Array, Object],
|
||||
default: () => ''
|
||||
},
|
||||
match: {
|
||||
type: String,
|
||||
default: 'exact'
|
||||
},
|
||||
attr: {
|
||||
type: Object,
|
||||
default: () => ({})
|
||||
}
|
||||
},
|
||||
data() {
|
||||
return {
|
||||
loading: true,
|
||||
type: 'string'
|
||||
}
|
||||
},
|
||||
computed: {
|
||||
iValue: {
|
||||
get() {
|
||||
const multipleTypes = ['array', 'select']
|
||||
let newValue = this.value
|
||||
if (multipleTypes.includes(this.type)) {
|
||||
if (!Array.isArray(this.value)) {
|
||||
newValue = []
|
||||
}
|
||||
} else if (this.type === 'bool') {
|
||||
newValue = !!this.value
|
||||
} else {
|
||||
if (Array.isArray(this.value)) {
|
||||
newValue = ''
|
||||
} else {
|
||||
newValue = this.value.toString()
|
||||
}
|
||||
}
|
||||
if (this.value !== newValue) {
|
||||
this.handleInput(newValue)
|
||||
}
|
||||
return newValue
|
||||
}
|
||||
}
|
||||
},
|
||||
watch: {
|
||||
match() {
|
||||
this.changeValueType()
|
||||
},
|
||||
attr: {
|
||||
handler() {
|
||||
this.changeValueType()
|
||||
},
|
||||
deep: true
|
||||
}
|
||||
},
|
||||
mounted() {
|
||||
this.changeValueType()
|
||||
},
|
||||
methods: {
|
||||
handleInput(value) {
|
||||
this.$emit('input', value)
|
||||
},
|
||||
changeValueType() {
|
||||
this.loading = true
|
||||
this.type = this.getType()
|
||||
this.$nextTick(() => {
|
||||
this.loading = false
|
||||
})
|
||||
},
|
||||
getType() {
|
||||
const attrType = this.attr.type || 'str'
|
||||
this.$log.debug('Value field attr type: ', attrType, this.attr, this.match)
|
||||
if (attrType === 'm2m') {
|
||||
return 'select'
|
||||
} else if (attrType === 'bool') {
|
||||
return 'bool'
|
||||
} else if (attrType === 'select') {
|
||||
return 'select'
|
||||
}
|
||||
if (['in', 'ip_in'].includes(this.match)) {
|
||||
return 'array'
|
||||
} else {
|
||||
return 'string'
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
</script>
|
||||
|
||||
<style scoped>
|
||||
|
||||
</style>
|
||||
@@ -0,0 +1,90 @@
|
||||
<template>
|
||||
<span v-if="attr.type === 'bool'">
|
||||
<i v-if="value" class="fa fa-check text-primary" />
|
||||
<i v-else class="fa fa-times text-danger" />
|
||||
</span>
|
||||
<span v-else :title="value">
|
||||
{{ value }}
|
||||
</span>
|
||||
</template>
|
||||
|
||||
<script>
|
||||
import BaseFormatter from '@/components/TableFormatters/base.vue'
|
||||
import { setUrlParam } from '@/utils/common'
|
||||
|
||||
export default {
|
||||
name: 'ValueFormatter',
|
||||
extends: BaseFormatter,
|
||||
props: {
|
||||
formatterArgsDefault: {
|
||||
type: Object,
|
||||
default() {
|
||||
return {
|
||||
attrs: {}
|
||||
}
|
||||
}
|
||||
}
|
||||
},
|
||||
data() {
|
||||
const formatterArgs = Object.assign(this.formatterArgsDefault, this.col.formatterArgs)
|
||||
return {
|
||||
formatterArgs: formatterArgs,
|
||||
loading: true,
|
||||
attr: {},
|
||||
value: ''
|
||||
}
|
||||
},
|
||||
watch: {
|
||||
cellValue: {
|
||||
handler() {
|
||||
this.getValue()
|
||||
},
|
||||
deep: true
|
||||
},
|
||||
formatterArgs: {
|
||||
handler() {
|
||||
this.getValue()
|
||||
},
|
||||
deep: true
|
||||
},
|
||||
row: {
|
||||
handler() {
|
||||
this.getValue()
|
||||
},
|
||||
deep: true
|
||||
}
|
||||
},
|
||||
mounted() {
|
||||
setTimeout(() => {
|
||||
this.getValue()
|
||||
}, 100)
|
||||
},
|
||||
methods: {
|
||||
async getValue() {
|
||||
this.attr = this.formatterArgs.attrs.find(attr => attr.name === this.row.name)
|
||||
const match = this.row.match
|
||||
this.$log.debug('ValueFormatter: ', this.attr, this.row.name)
|
||||
if (this.attr.type === 'm2m') {
|
||||
const url = setUrlParam(this.attr.el.url, 'ids', this.cellValue.join(','))
|
||||
const data = await this.$axios.get(url) || []
|
||||
if (data.length > 0) {
|
||||
const displayField = this.attr.el.displayField || 'name'
|
||||
this.value = data.map(item => item[displayField]).join(', ')
|
||||
}
|
||||
} else if (this.attr.type === 'select') {
|
||||
const options = this.attr.el.options || []
|
||||
const items = options.filter(item => this.cellValue.includes(item.value))
|
||||
this.value = items.map(item => item.label).join(', ')
|
||||
} else if (['in', 'ip_in'].includes(match)) {
|
||||
this.value = this.cellValue.join(', ')
|
||||
} else {
|
||||
this.value = this.cellValue
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
</script>
|
||||
|
||||
<style scoped>
|
||||
|
||||
</style>
|
||||
25
src/components/FormFields/JSONManyToManySelect/const.js
Normal file
25
src/components/FormFields/JSONManyToManySelect/const.js
Normal file
@@ -0,0 +1,25 @@
|
||||
import i18n from '@/i18n/i18n'
|
||||
|
||||
export const strMatchValues = ['exact', 'not', 'in', 'contains', 'startswith', 'endswith', 'regex']
|
||||
export const typeMatchMapper = {
|
||||
str: strMatchValues,
|
||||
bool: ['exact', 'not'],
|
||||
m2m: ['m2m'],
|
||||
ip: [...strMatchValues, 'ip_in'],
|
||||
int: [...strMatchValues, 'gte', 'lte'],
|
||||
select: ['in']
|
||||
}
|
||||
|
||||
export const attrMatchOptions = [
|
||||
{ label: i18n.t('common.Equal'), value: 'exact' },
|
||||
{ label: i18n.t('common.NotEqual'), value: 'not' },
|
||||
{ label: i18n.t('common.MatchIn'), value: 'in' },
|
||||
{ label: i18n.t('common.Contains'), value: 'contains' },
|
||||
{ label: i18n.t('common.Startswith'), value: 'startswith' },
|
||||
{ label: i18n.t('common.Endswith'), value: 'endswith' },
|
||||
{ label: i18n.t('common.Regex'), value: 'regex' },
|
||||
{ label: i18n.t('common.BelongTo'), value: 'm2m' },
|
||||
{ label: i18n.t('common.IPMatch'), value: 'ip_in' },
|
||||
{ label: i18n.t('common.GreatEqualThan'), value: 'gte' },
|
||||
{ label: i18n.t('common.LessEqualThan'), value: 'lte' }
|
||||
]
|
||||
240
src/components/FormFields/JSONManyToManySelect/index.vue
Normal file
240
src/components/FormFields/JSONManyToManySelect/index.vue
Normal file
@@ -0,0 +1,240 @@
|
||||
<template>
|
||||
<div>
|
||||
<el-radio-group v-model="iValue.type" @input="handleTypeChange">
|
||||
<el-radio v-for="tp of types" :key="tp.name" :label="tp.name">
|
||||
{{ tp.label }}
|
||||
</el-radio>
|
||||
</el-radio-group>
|
||||
<Select2 v-if="iValue.type === 'ids'" v-model="ids" v-bind="select2" @change="onChangeEmit" />
|
||||
<div v-if="iValue.type === 'attrs'">
|
||||
<DataTable :config="tableConfig" class="attr-list" />
|
||||
<div class="actions">
|
||||
<el-button size="mini" type="primary" @click="handleAttrAdd">
|
||||
{{ $t('common.Add') }}
|
||||
</el-button>
|
||||
<span style="padding-left: 10px; font-size: 13px">
|
||||
<span class="help-tips; ">{{ $t('common.MatchedCount') }}:</span>
|
||||
<a class="text-link" style="padding: 0 5px;" @click="showAttrMatchTable">{{ attrMatchCount }}</a>
|
||||
</span>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<AttrFormDialog
|
||||
v-if="attrFormVisible"
|
||||
:attrs="attrs"
|
||||
:attrs-added="attrsAdded"
|
||||
:form="attrForm"
|
||||
:visible.sync="attrFormVisible"
|
||||
@confirm="handleAttrDialogConfirm"
|
||||
/>
|
||||
<AttrMatchResultDialog
|
||||
v-if="attrMatchTableVisible"
|
||||
:attrs="attrs"
|
||||
:url="attrMatchTableUrl"
|
||||
:visible.sync="attrMatchTableVisible"
|
||||
/>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<script>
|
||||
import Select2 from '../Select2.vue'
|
||||
import DataTable from '@/components/DataTable/index.vue'
|
||||
import ValueFormatter from './ValueFormatter.vue'
|
||||
import AttrFormDialog from './AttrFormDialog.vue'
|
||||
import AttrMatchResultDialog from './AttrMatchResultDialog.vue'
|
||||
import { setUrlParam } from '@/utils/common'
|
||||
import { attrMatchOptions } from './const'
|
||||
import { toM2MJsonParams } from '@/utils/jms'
|
||||
|
||||
export default {
|
||||
name: 'JSONManyToManySelect',
|
||||
components: { AttrFormDialog, DataTable, Select2, AttrMatchResultDialog },
|
||||
props: {
|
||||
value: {
|
||||
type: Object,
|
||||
default: () => {
|
||||
return {
|
||||
type: 'all'
|
||||
}
|
||||
}
|
||||
},
|
||||
select2: {
|
||||
type: Object,
|
||||
required: true
|
||||
},
|
||||
attrs: {
|
||||
type: Array,
|
||||
default: () => ([])
|
||||
},
|
||||
resource: {
|
||||
type: String,
|
||||
default: ''
|
||||
},
|
||||
attrTableColumns: {
|
||||
type: Array,
|
||||
default: () => (['name'])
|
||||
}
|
||||
},
|
||||
data() {
|
||||
const tableFormatter = (colName) => {
|
||||
return (row, col, cellValue) => {
|
||||
const value = cellValue
|
||||
switch (colName) {
|
||||
case 'name':
|
||||
return this.attrs.find(attr => attr.name === value)?.label || value
|
||||
case 'match':
|
||||
return attrMatchOptions.find(opt => opt.value === value).label || value
|
||||
case 'value':
|
||||
return Array.isArray(value) ? value.join(', ') : value
|
||||
default:
|
||||
return value
|
||||
}
|
||||
}
|
||||
}
|
||||
return {
|
||||
iValue: Object.assign({ type: 'all' }, this.value),
|
||||
attrFormVisible: false,
|
||||
attrForm: {},
|
||||
attrMatchCount: 0,
|
||||
attrMatchTableVisible: false,
|
||||
attrMatchTableUrl: '',
|
||||
ids: this.value.ids || [],
|
||||
editIndex: -1,
|
||||
types: [
|
||||
{ name: 'all', label: this.$t('common.All') + this.resource },
|
||||
{ name: 'ids', label: this.$t('common.Spec') + this.resource },
|
||||
{ name: 'attrs', label: this.$t('common.SelectByAttr') }
|
||||
],
|
||||
tableConfig: {
|
||||
columns: [
|
||||
{ prop: 'name', label: this.$t('common.AttrName'), formatter: tableFormatter('name') },
|
||||
{ prop: 'match', label: this.$t('common.Match'), formatter: tableFormatter('match') },
|
||||
{ prop: 'value', label: this.$t('common.AttrValue'), formatter: ValueFormatter, formatterArgs: { attrs: this.attrs }},
|
||||
{ prop: 'action', label: this.$t('common.Action'), align: 'center', width: '120px', formatter: (row, col, cellValue, index) => {
|
||||
return (
|
||||
<div className='input-button'>
|
||||
<el-button
|
||||
icon='el-icon-edit'
|
||||
size='mini'
|
||||
style={{ 'flexShrink': 0 }}
|
||||
type='primary'
|
||||
onClick={this.handleAttrEdit({ row, col, cellValue, index })}
|
||||
/>
|
||||
<el-button
|
||||
icon='el-icon-minus'
|
||||
size='mini'
|
||||
style={{ 'flexShrink': 0 }}
|
||||
type='danger'
|
||||
onClick={this.handleAttrDelete({ row, col, cellValue, index })}
|
||||
/>
|
||||
</div>
|
||||
)
|
||||
} }
|
||||
],
|
||||
totalData: this.value.attrs || [],
|
||||
hasPagination: false
|
||||
}
|
||||
}
|
||||
},
|
||||
computed: {
|
||||
attrsAdded() {
|
||||
return this.tableConfig.totalData.map(item => item.name)
|
||||
}
|
||||
},
|
||||
watch: {
|
||||
attrFormVisible(val) {
|
||||
if (!val) {
|
||||
this.getAttrsCount()
|
||||
}
|
||||
}
|
||||
},
|
||||
mounted() {
|
||||
if (this.value.type === 'attrs') {
|
||||
this.getAttrsCount()
|
||||
}
|
||||
this.$emit('input', this.iValue)
|
||||
},
|
||||
methods: {
|
||||
showAttrMatchTable() {
|
||||
const [key, value] = this.getAttrFilterKey()
|
||||
this.attrMatchTableUrl = setUrlParam(this.select2.url, key, value)
|
||||
this.attrMatchTableVisible = true
|
||||
},
|
||||
getAttrFilterKey() {
|
||||
if (this.tableConfig.totalData.length === 0) return ''
|
||||
let attrFilter = { type: 'attrs', attrs: this.tableConfig.totalData }
|
||||
attrFilter = toM2MJsonParams(attrFilter)
|
||||
return attrFilter
|
||||
},
|
||||
getAttrsCount() {
|
||||
const attrFilter = this.getAttrFilterKey()
|
||||
if (!attrFilter) {
|
||||
this.attrMatchCount = 0
|
||||
return
|
||||
}
|
||||
const [key, value] = attrFilter
|
||||
let url = setUrlParam(this.select2.url, key, value)
|
||||
url = setUrlParam(url, 'limit', 1)
|
||||
return this.$axios.get(url).then(res => {
|
||||
this.attrMatchCount = res.count
|
||||
})
|
||||
},
|
||||
handleAttrEdit({ row, index }) {
|
||||
return () => {
|
||||
this.attrForm = Object.assign({ index }, row)
|
||||
this.editIndex = index
|
||||
this.attrFormVisible = true
|
||||
}
|
||||
},
|
||||
handleAttrDelete({ index }) {
|
||||
return () => {
|
||||
this.tableConfig.totalData.splice(index, 1)
|
||||
this.getAttrsCount()
|
||||
}
|
||||
},
|
||||
handleAttrAdd() {
|
||||
this.attrForm = {}
|
||||
this.editIndex = -1
|
||||
this.attrFormVisible = true
|
||||
},
|
||||
onChangeEmit() {
|
||||
const tp = this.iValue.type
|
||||
this.handleTypeChange(tp)
|
||||
},
|
||||
handleTypeChange(val) {
|
||||
switch (val) {
|
||||
case 'ids':
|
||||
this.$emit('input', { type: 'ids', ids: this.ids })
|
||||
break
|
||||
case 'attrs':
|
||||
this.$emit('input', { type: 'attrs', attrs: this.tableConfig.totalData })
|
||||
break
|
||||
default:
|
||||
this.$emit('input', { type: 'all' })
|
||||
break
|
||||
}
|
||||
},
|
||||
handleAttrDialogConfirm(form) {
|
||||
if (this.editIndex > -1) {
|
||||
this.tableConfig.totalData.splice(this.editIndex, 1)
|
||||
}
|
||||
const allAttrs = this.tableConfig.totalData
|
||||
// 因为可能 attr 的 name 会重复,所以需要先删除再添加
|
||||
const setIndex = allAttrs.findIndex(attr => attr.name === form.name)
|
||||
if (setIndex === -1) {
|
||||
allAttrs.push(Object.assign({}, form))
|
||||
} else {
|
||||
allAttrs.splice(setIndex, 1, Object.assign({}, form))
|
||||
}
|
||||
this.attrFormVisible = false
|
||||
this.onChangeEmit()
|
||||
}
|
||||
}
|
||||
}
|
||||
</script>
|
||||
|
||||
<style lang="scss" scoped>
|
||||
.attr-list {
|
||||
width: 99%;
|
||||
}
|
||||
</style>
|
||||
@@ -2,8 +2,9 @@
|
||||
<div class="json-editor">
|
||||
<JsonEditor
|
||||
v-model="resultInfo"
|
||||
:show-btns="false"
|
||||
:mode="'code'"
|
||||
:show-btns="false"
|
||||
:class="{resize: resize === 'vertical'}"
|
||||
@json-change="onJsonChange"
|
||||
@json-save="onJsonSave"
|
||||
@has-error="onError"
|
||||
@@ -18,8 +19,15 @@ export default {
|
||||
components: { JsonEditor },
|
||||
props: {
|
||||
value: {
|
||||
type: [String, Object, Array],
|
||||
default: () => ({})
|
||||
},
|
||||
resize: {
|
||||
type: String,
|
||||
default: () => ''
|
||||
validator: (value) => {
|
||||
return ['none', 'vertical'].indexOf(value) !== -1
|
||||
},
|
||||
default: 'vertical'
|
||||
}
|
||||
},
|
||||
data() {
|
||||
@@ -29,7 +37,7 @@ export default {
|
||||
}
|
||||
},
|
||||
created() {
|
||||
this.resultInfo = JSON.parse(this.value)
|
||||
this.resultInfo = typeof this.value === 'string' ? JSON.parse(this.value) : this.value
|
||||
},
|
||||
methods: {
|
||||
// 数据改变
|
||||
@@ -38,35 +46,46 @@ export default {
|
||||
},
|
||||
// 保存
|
||||
onJsonSave(value) {
|
||||
this.resultInfo = value
|
||||
this.resultInfo = typeof value === 'string' ? JSON.parse(value) : value
|
||||
this.hasJsonFlag = true
|
||||
setTimeout(() => {
|
||||
this.$emit('change', JSON.stringify(this.resultInfo))
|
||||
this.$emit('change', this.resultInfo)
|
||||
}, 500)
|
||||
},
|
||||
onError: _.debounce(function(value) {
|
||||
this.$message.error(this.$t('common.FormatError'))
|
||||
}, 1100)
|
||||
this.$message.error(this.$tc('common.FormatError'))
|
||||
}, 1500)
|
||||
}
|
||||
}
|
||||
</script>
|
||||
|
||||
<style lang="scss" scoped>
|
||||
@import "~@/styles/variables.scss";
|
||||
|
||||
.json-editor {
|
||||
&>>> .jsoneditor {
|
||||
.resize {
|
||||
& > > > .jsoneditor {
|
||||
resize: vertical;
|
||||
cursor: s-resize;
|
||||
}
|
||||
}
|
||||
& > > > .jsoneditor {
|
||||
border: 1px solid #e5e6e7;
|
||||
}
|
||||
&>>> .jsoneditor-compact {
|
||||
|
||||
& > > > .jsoneditor-compact {
|
||||
display: none;
|
||||
}
|
||||
&>>> .jsoneditor-modes {
|
||||
|
||||
& > > > .jsoneditor-modes {
|
||||
display: none;
|
||||
}
|
||||
&>>> .jsoneditor-poweredBy {
|
||||
|
||||
& > > > .jsoneditor-poweredBy {
|
||||
display: none;
|
||||
}
|
||||
&>>> .jsoneditor-menu {
|
||||
|
||||
& > > > .jsoneditor-menu {
|
||||
background: var(--color-primary);
|
||||
border-bottom: 1px solid var(--color-primary);
|
||||
}
|
||||
|
||||
97
src/components/FormFields/ListField.vue
Normal file
97
src/components/FormFields/ListField.vue
Normal file
@@ -0,0 +1,97 @@
|
||||
<template>
|
||||
<div>
|
||||
<div v-for="(item, index) in value || []" :key="index" class="value-item">
|
||||
<el-input :value="item" class="input-z" @input="updateValue($event, index)" />
|
||||
<div class="input-button">
|
||||
<el-button
|
||||
:disabled="disableDelete(item)"
|
||||
icon="el-icon-minus"
|
||||
size="mini"
|
||||
style="flex-shrink: 0;"
|
||||
type="danger"
|
||||
@click="handleDelete(index)"
|
||||
/>
|
||||
<el-button
|
||||
:disabled="disableAdd(item, index)"
|
||||
icon="el-icon-plus"
|
||||
size="mini"
|
||||
style="flex-shrink: 0;"
|
||||
type="primary"
|
||||
@click="handleAdd(index)"
|
||||
/>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<script>
|
||||
export default {
|
||||
name: 'ListField',
|
||||
props: {
|
||||
value: {
|
||||
type: [Array, String],
|
||||
default: () => ([])
|
||||
}
|
||||
},
|
||||
data() {
|
||||
return {
|
||||
}
|
||||
},
|
||||
computed: {
|
||||
},
|
||||
mounted() {
|
||||
const value = this.value
|
||||
if (!value || !Array.isArray(value) || value.length === 0) {
|
||||
this.$emit('input', [''])
|
||||
}
|
||||
},
|
||||
methods: {
|
||||
updateValue(v, index) {
|
||||
const value = this.value
|
||||
value[index] = v
|
||||
this.$emit('input', value)
|
||||
},
|
||||
disableDelete() {
|
||||
return false
|
||||
},
|
||||
disableAdd() {
|
||||
return false
|
||||
},
|
||||
handleAdd(index) {
|
||||
const value = this.value
|
||||
value.splice(index + 1, 0, '')
|
||||
this.$emit('input', value)
|
||||
},
|
||||
handleDelete(index) {
|
||||
const value = this.value
|
||||
value.splice(index, 1)
|
||||
this.$emit('input', value)
|
||||
}
|
||||
}
|
||||
}
|
||||
</script>
|
||||
|
||||
<style lang="scss" scoped>
|
||||
|
||||
.input-z {
|
||||
flex-shrink: 1;
|
||||
width: calc(100% - 80px) !important;
|
||||
}
|
||||
|
||||
.value-item {
|
||||
display: flex;
|
||||
margin: 4px 0;
|
||||
}
|
||||
|
||||
.input-button {
|
||||
display: flex;
|
||||
margin-left: 20px;
|
||||
margin-top: 4px;
|
||||
}
|
||||
|
||||
.input-button ::v-deep .el-button.el-button--mini {
|
||||
height: 25px;
|
||||
padding: 5px;
|
||||
}
|
||||
|
||||
</style>
|
||||
105
src/components/FormFields/NestedObjectSelect2.vue
Normal file
105
src/components/FormFields/NestedObjectSelect2.vue
Normal file
@@ -0,0 +1,105 @@
|
||||
<template>
|
||||
<Select2
|
||||
v-model="iValue"
|
||||
:multiple="multiple"
|
||||
v-bind="attrsWithoutValue"
|
||||
@change="onChange"
|
||||
@change-options="onChangeOptions"
|
||||
/>
|
||||
</template>
|
||||
|
||||
<script>
|
||||
import Select2 from './Select2'
|
||||
|
||||
export default {
|
||||
name: 'NestedObjectSelect2',
|
||||
components: {
|
||||
Select2
|
||||
},
|
||||
props: {
|
||||
value: {
|
||||
type: [Array, String, Number, Boolean, Object],
|
||||
default: () => ([])
|
||||
},
|
||||
multiple: {
|
||||
type: Boolean,
|
||||
default: true
|
||||
},
|
||||
// 自定义label字段的name
|
||||
customLabelKeyName: {
|
||||
type: String,
|
||||
default: 'name'
|
||||
}
|
||||
},
|
||||
data() {
|
||||
return {}
|
||||
},
|
||||
computed: {
|
||||
attrsWithoutValue() {
|
||||
const attrs = Object.assign({}, this.$attrs)
|
||||
delete attrs.value
|
||||
return attrs
|
||||
},
|
||||
iValue: {
|
||||
set(val) {
|
||||
const value = this.valuesToObjects(val)
|
||||
this.$log.debug('set iValue', value)
|
||||
this.$emit('input', value)
|
||||
},
|
||||
get() {
|
||||
const value = this.objectsToValues(this.value)
|
||||
return value
|
||||
}
|
||||
}
|
||||
},
|
||||
methods: {
|
||||
onChange(val) {
|
||||
val = this.valuesToObjects(val)
|
||||
this.$log.debug('onChange .... ', val)
|
||||
this.$emit('change', val)
|
||||
},
|
||||
onChangeOptions(val) {
|
||||
val = this.valuesToObjects(val)
|
||||
this.$log.debug('onChangeOptions', val)
|
||||
this.$emit('changeOptions', val)
|
||||
},
|
||||
valuesToObjects(values) {
|
||||
let value = values
|
||||
if (!this.multiple && !Array.isArray(value)) {
|
||||
value = [value]
|
||||
}
|
||||
value = value.map(v => {
|
||||
// uuid v4
|
||||
const uuid = /^[0-9A-F]{8}-[0-9A-F]{4}-[4][0-9A-F]{3}-[89AB][0-9A-F]{3}-[0-9A-F]{12}$/i
|
||||
return typeof v === 'object' ? v
|
||||
: this.$attrs?.allowCreate && !uuid.test(v) ? { [this.customLabelKeyName]: v } : { pk: v }
|
||||
})
|
||||
if (!this.multiple) {
|
||||
value = value[0]
|
||||
}
|
||||
return value
|
||||
},
|
||||
objectsToValues(objects) {
|
||||
let val = objects
|
||||
if (!this.multiple) {
|
||||
val = [val]
|
||||
}
|
||||
val = val.map((v) => {
|
||||
if (v && typeof v === 'object') {
|
||||
return v.pk || v.id || (this.$attrs?.allowCreate ? (v?.[this.customLabelKeyName] + ':' + v?.value) : '')
|
||||
} else {
|
||||
return v
|
||||
}
|
||||
})
|
||||
if (!this.multiple) {
|
||||
val = val[0]
|
||||
}
|
||||
return val
|
||||
}
|
||||
}
|
||||
}
|
||||
</script>
|
||||
|
||||
<style scoped>
|
||||
|
||||
</style>
|
||||
80
src/components/FormFields/PhoneInput.vue
Normal file
80
src/components/FormFields/PhoneInput.vue
Normal file
@@ -0,0 +1,80 @@
|
||||
<template>
|
||||
<div>
|
||||
<el-input v-model="rawValue.phone" :placeholder="$tc('users.inputPhone')" @input="OnInputChange">
|
||||
<el-select
|
||||
slot="prepend"
|
||||
:value="rawValue.code"
|
||||
:placeholder="$tc('common.Select')"
|
||||
style="width: 90px;"
|
||||
@change="OnChange"
|
||||
>
|
||||
<el-option
|
||||
v-for="country in countries"
|
||||
:key="country.value"
|
||||
:label="country.value"
|
||||
:value="country.value"
|
||||
style="width: 200px;"
|
||||
>
|
||||
<span style="float: left">{{ country.name }}</span>
|
||||
<span style="float: right; font-size: 13px">{{ country.value }}</span>
|
||||
</el-option>
|
||||
</el-select>
|
||||
</el-input>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<script>
|
||||
|
||||
export default {
|
||||
name: 'PhoneInput',
|
||||
props: {
|
||||
value: {
|
||||
type: [Object, String],
|
||||
default: () => ({ 'code': '+86', 'phone': '' })
|
||||
}
|
||||
},
|
||||
data() {
|
||||
return {
|
||||
rawValue: {},
|
||||
countries: [
|
||||
{ name: 'China(中国)', value: '+86' },
|
||||
{ name: 'HongKong(中国香港)', value: '+852' },
|
||||
{ name: 'Macao(中国澳门)', value: '+853' },
|
||||
{ name: 'Taiwan(中国台湾)', value: '+886' },
|
||||
{ name: 'America(America)', value: '+1' },
|
||||
{ name: 'Russia(Россия)', value: '+7' },
|
||||
{ name: 'France(français)', value: '+33' },
|
||||
{ name: 'Britain(Britain)', value: '+44' },
|
||||
{ name: 'Germany(Deutschland)', value: '+49' },
|
||||
{ name: 'Japan(日本)', value: '+81' },
|
||||
{ name: 'Korea(한국)', value: '+82' },
|
||||
{ name: 'India(भारत)', value: '+91' }
|
||||
]
|
||||
}
|
||||
},
|
||||
computed: {
|
||||
fullPhone() {
|
||||
if (!this.rawValue.phone) {
|
||||
return ''
|
||||
}
|
||||
return `${this.rawValue.code}${this.rawValue.phone}`
|
||||
}
|
||||
},
|
||||
mounted() {
|
||||
this.rawValue = this.value || { code: '+86', phone: '' }
|
||||
this.$emit('input', this.fullPhone)
|
||||
},
|
||||
methods: {
|
||||
OnChange(countryCode) {
|
||||
this.rawValue.code = countryCode
|
||||
this.OnInputChange()
|
||||
},
|
||||
OnInputChange() {
|
||||
this.$emit('input', this.fullPhone)
|
||||
}
|
||||
}
|
||||
}
|
||||
</script>
|
||||
|
||||
<style lang="scss" scoped>
|
||||
</style>
|
||||
@@ -0,0 +1,116 @@
|
||||
<template>
|
||||
<Dialog
|
||||
v-if="$attrs.visible"
|
||||
:close-on-click-modal="false"
|
||||
:destroy-on-close="true"
|
||||
:show-cancel="false"
|
||||
:show-confirm="false"
|
||||
:title="$tc('assets.PlatformProtocolConfig') + ':' + protocol.name"
|
||||
class="setting-dialog"
|
||||
v-bind="$attrs"
|
||||
width="70%"
|
||||
v-on="$listeners"
|
||||
>
|
||||
<el-alert v-if="disabled && platformDetail" style="margin-bottom: 10px" type="success">
|
||||
{{ $t('assets.InheritPlatformConfig') }}
|
||||
<el-link :href="platformDetail" class="link-more" target="_blank">
|
||||
{{ $t('common.View') }}
|
||||
</el-link>
|
||||
<i class="fa fa-external-link" />
|
||||
</el-alert>
|
||||
<AutoDataForm
|
||||
:disabled="disabled"
|
||||
:form="form"
|
||||
class="data-form"
|
||||
v-bind="config"
|
||||
@submit="onSubmit"
|
||||
/>
|
||||
</Dialog>
|
||||
</template>
|
||||
|
||||
<script>
|
||||
import { AutoDataForm, Dialog } from '@/components'
|
||||
import JsonEditor from '@/components/FormFields/JsonEditor.vue'
|
||||
|
||||
export default {
|
||||
name: 'ProtocolSetting',
|
||||
components: {
|
||||
Dialog,
|
||||
AutoDataForm
|
||||
},
|
||||
props: {
|
||||
protocol: {
|
||||
type: Object,
|
||||
default: () => ({})
|
||||
},
|
||||
disabled: {
|
||||
type: Boolean,
|
||||
default: false
|
||||
}
|
||||
},
|
||||
data() {
|
||||
const vm = this
|
||||
const platform = this.$route.query.platform
|
||||
return {
|
||||
loading: true,
|
||||
form: this.protocol,
|
||||
platformDetail: platform ? '#/console/assets/platforms/' + platform : '',
|
||||
config: {
|
||||
hasSaveContinue: false,
|
||||
hasButtons: !this.disabled,
|
||||
url: '/api/v1/assets/protocol-settings/?name=' + this.protocol.name,
|
||||
fields: [
|
||||
[vm.$t('common.Basic'), [
|
||||
'primary', 'required', 'default', 'public'
|
||||
]],
|
||||
[vm.$t('common.Advanced'), ['setting']]
|
||||
],
|
||||
fieldsMeta: {
|
||||
setting: {
|
||||
fields: '__all__',
|
||||
fieldsMeta: {
|
||||
username_selector: {
|
||||
hidden: (formValue) => formValue['autofill'] !== 'basic'
|
||||
},
|
||||
password_selector: {
|
||||
hidden: (formValue) => formValue['autofill'] !== 'basic'
|
||||
},
|
||||
submit_selector: {
|
||||
hidden: (formValue) => formValue['autofill'] !== 'basic'
|
||||
},
|
||||
script: {
|
||||
component: JsonEditor,
|
||||
hidden: (formValue) => formValue['autofill'] !== 'script'
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
},
|
||||
methods: {
|
||||
onSubmit(form) {
|
||||
this.protocol = Object.assign(this.protocol, form)
|
||||
this.$emit('update:visible', false)
|
||||
this.$emit('confirm', this.protocol)
|
||||
}
|
||||
}
|
||||
}
|
||||
</script>
|
||||
|
||||
<style lang="scss" scoped>
|
||||
.data-form > > > .el-form-item.form-buttons {
|
||||
padding-top: 10px;
|
||||
margin-bottom: 0;
|
||||
}
|
||||
|
||||
.setting-dialog > > > .el-dialog__body {
|
||||
padding-top: 10px;
|
||||
}
|
||||
|
||||
.link-more {
|
||||
font-size: 10px;
|
||||
border-bottom: solid 1px;
|
||||
color: inherit;
|
||||
}
|
||||
</style>
|
||||
366
src/components/FormFields/ProtocolSelector/index.vue
Normal file
366
src/components/FormFields/ProtocolSelector/index.vue
Normal file
@@ -0,0 +1,366 @@
|
||||
<template>
|
||||
<div v-if="!loading" :class="showSetting ? 'show-setting' : 'hide-setting'">
|
||||
<div v-for="(item, index) in items" :key="item.name" class="protocol-item">
|
||||
<el-input
|
||||
v-model="item.port"
|
||||
:class="isPortReadonly(item) ? '' : 'input-with-select'"
|
||||
:placeholder="portPlaceholder"
|
||||
:readonly="isPortReadonly(item)"
|
||||
:title="isPortReadonly(item) ? '端口由 URL 指定' : ''"
|
||||
v-bind="$attrs"
|
||||
>
|
||||
<template #prepend>
|
||||
<el-select
|
||||
:disabled="disableSelect(item)"
|
||||
:value="item.display_name ? item.display_name : item.name"
|
||||
class="prepend"
|
||||
@change="handleProtocolChange($event, item)"
|
||||
>
|
||||
<el-option
|
||||
v-for="p of remainProtocols"
|
||||
:key="p.name"
|
||||
:label="p.name"
|
||||
:value="p.name"
|
||||
/>
|
||||
</el-select>
|
||||
</template>
|
||||
<template #append>
|
||||
<el-button
|
||||
v-if="showSetting(item)"
|
||||
icon="el-icon-setting"
|
||||
@click="onSettingClick(item)"
|
||||
/>
|
||||
</template>
|
||||
</el-input>
|
||||
<div v-if="!readonly" class="input-button">
|
||||
<el-button
|
||||
:disabled="disableDelete(item)"
|
||||
icon="el-icon-minus"
|
||||
size="mini"
|
||||
style="flex-shrink: 0;"
|
||||
type="danger"
|
||||
@click="handleDelete(index)"
|
||||
/>
|
||||
<el-button
|
||||
v-if="index === items.length - 1"
|
||||
:disabled="disableAdd(item, index)"
|
||||
icon="el-icon-plus"
|
||||
size="mini"
|
||||
style="flex-shrink: 0;"
|
||||
type="primary"
|
||||
@click="handleAdd(index)"
|
||||
/>
|
||||
</div>
|
||||
</div>
|
||||
<el-button
|
||||
v-if="items.length === 0"
|
||||
icon="el-icon-plus"
|
||||
size="mini"
|
||||
style="flex-shrink: 0;"
|
||||
type="primary"
|
||||
@click="handleAdd(0)"
|
||||
/>
|
||||
<ProtocolSettingDialog
|
||||
v-if="showDialog"
|
||||
:disabled="settingReadonly || readonly"
|
||||
:protocol="currentProtocol"
|
||||
:visible.sync="showDialog"
|
||||
@confirm="handleSettingConfirm"
|
||||
/>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<script>
|
||||
import ProtocolSettingDialog from './ProtocolSettingDialog'
|
||||
|
||||
export default {
|
||||
components: {
|
||||
ProtocolSettingDialog
|
||||
},
|
||||
props: {
|
||||
value: {
|
||||
type: [String, Array],
|
||||
default: () => []
|
||||
},
|
||||
title: {
|
||||
type: String,
|
||||
default: ''
|
||||
},
|
||||
choices: {
|
||||
type: Array,
|
||||
default: () => ([])
|
||||
},
|
||||
readonly: {
|
||||
// 这个是在详情中,不可编辑,包括所有
|
||||
type: Boolean,
|
||||
default: false
|
||||
},
|
||||
settingReadonly: {
|
||||
// 这个是在资产添加时设置协议使用,不能修改 setting
|
||||
type: Boolean,
|
||||
default: false
|
||||
},
|
||||
showSetting: {
|
||||
type: Function,
|
||||
default: (item) => true
|
||||
},
|
||||
instance: {
|
||||
type: Object,
|
||||
default: () => ({})
|
||||
}
|
||||
},
|
||||
data() {
|
||||
return {
|
||||
name: '',
|
||||
items: [],
|
||||
currentProtocol: {},
|
||||
showDialog: false,
|
||||
loading: false
|
||||
}
|
||||
},
|
||||
computed: {
|
||||
selectedProtocolNames() {
|
||||
return this.items.map(item => item.name)
|
||||
},
|
||||
remainProtocols() {
|
||||
return this.choices.filter(proto => {
|
||||
return this.selectedProtocolNames.indexOf(proto.name) === -1
|
||||
})
|
||||
},
|
||||
portPlaceholder() {
|
||||
if (this.settingReadonly) {
|
||||
return this.$t('applications.port')
|
||||
} else {
|
||||
return this.$t('assets.DefaultPort')
|
||||
}
|
||||
},
|
||||
iChoices() {
|
||||
return this.choices.map(item => {
|
||||
delete item?.id
|
||||
return item
|
||||
})
|
||||
}
|
||||
},
|
||||
watch: {
|
||||
choices: {
|
||||
handler(value, oldValue) {
|
||||
if (value?.length === oldValue?.length) {
|
||||
return
|
||||
}
|
||||
this.loading = true
|
||||
setTimeout(() => {
|
||||
this.setDefaultItems(value)
|
||||
this.loading = false
|
||||
},)
|
||||
},
|
||||
deep: true,
|
||||
immediate: true
|
||||
},
|
||||
items: {
|
||||
handler(value) {
|
||||
if (this.settingReadonly) {
|
||||
value = value.map(i => {
|
||||
return { name: i.name, port: i.port }
|
||||
})
|
||||
}
|
||||
this.$emit('input', value)
|
||||
},
|
||||
immediate: true,
|
||||
deep: true
|
||||
},
|
||||
instance: {
|
||||
handler(value) {
|
||||
const port = this.getPortFromInstance(value)
|
||||
if (!port) {
|
||||
return
|
||||
}
|
||||
for (const item of this.items) {
|
||||
if (item['port_from_addr']) {
|
||||
item.port = port
|
||||
}
|
||||
}
|
||||
},
|
||||
deep: true,
|
||||
immediate: true
|
||||
}
|
||||
},
|
||||
mounted() {
|
||||
this.setDefaultItems(this.iChoices)
|
||||
this.$log.debug('Choices: ', this.choices)
|
||||
this.$log.debug('Value: ', this.value)
|
||||
this.$log.debug('Items: ', this.items)
|
||||
},
|
||||
methods: {
|
||||
getPortFromInstance(instance) {
|
||||
if (!instance) {
|
||||
return 0
|
||||
}
|
||||
let address = instance.address || ''
|
||||
if (address.indexOf('://') === -1) {
|
||||
address = `https://${address}`
|
||||
}
|
||||
const parse = require('url-parse')
|
||||
const path = parse(address)
|
||||
let port = path.port
|
||||
if (port < 0 || port > 65535) {
|
||||
port = 0
|
||||
}
|
||||
if (!port) {
|
||||
port = path.protocol === 'https:' ? 443 : 80
|
||||
}
|
||||
return port
|
||||
},
|
||||
handleSettingConfirm() {
|
||||
if (this.currentProtocol.primary) {
|
||||
const others = this.items
|
||||
.filter(item => item.name !== this.currentProtocol.name)
|
||||
.map(item => {
|
||||
item.primary = false
|
||||
return item
|
||||
})
|
||||
this.items = [this.currentProtocol, ...others]
|
||||
}
|
||||
if (this.currentProtocol.name === 'winrm') {
|
||||
if (this.currentProtocol.setting?.use_ssl) {
|
||||
this.currentProtocol.port = 5986
|
||||
} else {
|
||||
this.currentProtocol.port = 5985
|
||||
}
|
||||
}
|
||||
},
|
||||
handleDelete(index) {
|
||||
this.items = this.items.filter((value, i) => i !== index)
|
||||
},
|
||||
isRequired(item) {
|
||||
const full = this.iChoices.find(choice => {
|
||||
return choice.name === item.name
|
||||
})
|
||||
return full?.primary || full?.required
|
||||
},
|
||||
disableSelect(item) {
|
||||
return this.isRequired(item)
|
||||
},
|
||||
disableDelete(item) {
|
||||
if (this.items.length === 1) {
|
||||
return true
|
||||
}
|
||||
// 代表是设置平台
|
||||
if (!this.settingReadonly) {
|
||||
return false
|
||||
}
|
||||
return this.isRequired(item)
|
||||
},
|
||||
disableAdd(item) {
|
||||
return this.remainProtocols.length === 0 || !item.port
|
||||
},
|
||||
handleAdd(index) {
|
||||
this.items.push({ ...this.remainProtocols[0] })
|
||||
},
|
||||
handleProtocolChange(evt, item) {
|
||||
const selected = this.choices.find(item => item.name === evt)
|
||||
item.name = selected.name
|
||||
item.port = selected.port
|
||||
},
|
||||
isPortFormAddr(item) {
|
||||
return !!item['port_from_addr']
|
||||
},
|
||||
isPortReadonly(item) {
|
||||
return this.readonly || this.isPortFormAddr(item)
|
||||
},
|
||||
setPrimaryIfNeed(items) {
|
||||
// 如果没有设置主协议,设置第一个为主协议
|
||||
if (this.settingReadonly) {
|
||||
return items
|
||||
}
|
||||
const primaryProtocols = items.filter(item => item.primary)
|
||||
if (primaryProtocols.length === 0) {
|
||||
items[0].default = true
|
||||
items[0].public = true
|
||||
} else if (primaryProtocols.length > 1) {
|
||||
primaryProtocols.slice(1, primaryProtocols.length).forEach(item => {
|
||||
item.primary = false
|
||||
})
|
||||
}
|
||||
return items
|
||||
},
|
||||
setDefaultItems(choices) {
|
||||
let items = []
|
||||
const requiredItems = choices.filter(item => (item.required || item.primary))
|
||||
|
||||
if (this.value instanceof Array && this.value.length > 0) {
|
||||
const protocols = []
|
||||
this.value.forEach(item => {
|
||||
// 有默认值的情况下,设置为只读或者有id、有setting是平台
|
||||
if (!this.settingReadonly || (item?.id && item?.setting)) {
|
||||
protocols.push(item)
|
||||
} else {
|
||||
// 获取资产协议配置
|
||||
const assetDefaultItems = this.getAssetDefaultItems(item, choices)
|
||||
protocols.push(...assetDefaultItems)
|
||||
}
|
||||
})
|
||||
const notFound = requiredItems.filter(item => !protocols.find(p => p.name === item.name))
|
||||
protocols.push(...notFound)
|
||||
const allProtocolNames = protocols.map(item => item.name)
|
||||
items = protocols.filter(item => allProtocolNames.indexOf(item.name) !== -1)
|
||||
} else {
|
||||
const defaults = choices.filter(item => (item.required || item.primary || item.default))
|
||||
if (defaults.length === 0) {
|
||||
defaults.push(choices[0])
|
||||
}
|
||||
items = defaults
|
||||
}
|
||||
items = this.setPrimaryIfNeed(items)
|
||||
this.items = items
|
||||
},
|
||||
getAssetDefaultItems(item, choices) {
|
||||
const protocols = []
|
||||
const protocol = choices.find(i => i.name === item.name) || {}
|
||||
protocols.push({ ...protocol, ...item })
|
||||
return protocols
|
||||
},
|
||||
onSettingClick(item) {
|
||||
this.currentProtocol = item
|
||||
this.showDialog = true
|
||||
}
|
||||
}
|
||||
}
|
||||
</script>
|
||||
<style lang="scss" scoped>
|
||||
.el-select >>> .el-input__inner {
|
||||
width: 120px;
|
||||
text-overflow: ellipsis;
|
||||
white-space: nowrap;
|
||||
}
|
||||
|
||||
.input-with-select {
|
||||
flex-shrink: 1;
|
||||
width: calc(100% - 80px) !important;
|
||||
}
|
||||
|
||||
.input-with-select .el-input-group__prepend {
|
||||
background-color: #fff;
|
||||
}
|
||||
|
||||
.protocol-item {
|
||||
display: flex;
|
||||
margin: 5px 0;
|
||||
}
|
||||
|
||||
.input-button {
|
||||
margin-top: 4px;
|
||||
display: flex;
|
||||
margin-left: 20px
|
||||
}
|
||||
|
||||
.input-button ::v-deep .el-button.el-button--mini {
|
||||
height: 25px;
|
||||
padding: 5px;
|
||||
}
|
||||
|
||||
.el-input-group__append .el-button {
|
||||
font-size: 14px;
|
||||
color: #1a1a1a;
|
||||
padding: 9px 20px;
|
||||
}
|
||||
</style>
|
||||
5
src/components/FormFields/Secret.vue
Normal file
5
src/components/FormFields/Secret.vue
Normal file
@@ -0,0 +1,5 @@
|
||||
<template>
|
||||
<div>
|
||||
Todo: 抽象 Secret
|
||||
</div>
|
||||
</template>
|
||||
@@ -2,28 +2,29 @@
|
||||
<el-select
|
||||
ref="select"
|
||||
v-model="iValue"
|
||||
v-loading="!initialized"
|
||||
v-loadmore="loadMore"
|
||||
:clearable="clearable"
|
||||
:collapse-tags="collapseTags"
|
||||
:disabled="!!selectDisabled"
|
||||
:loading="!initialized"
|
||||
:multiple="multiple"
|
||||
:options="iOptions"
|
||||
:remote="remote"
|
||||
:remote-method="filterOptions"
|
||||
:multiple="multiple"
|
||||
:clearable="clearable"
|
||||
class="select2"
|
||||
filterable
|
||||
popper-append-to-body
|
||||
class="select2"
|
||||
:disabled="selectDisabled"
|
||||
v-bind="$attrs"
|
||||
@change="onChange"
|
||||
@visible-change="onVisibleChange"
|
||||
v-on="$listeners"
|
||||
@visible-change="onVisibleChange"
|
||||
>
|
||||
<el-option
|
||||
v-for="item in iOptions"
|
||||
:key="item.value"
|
||||
:disabled="checkDisabled(item)"
|
||||
:label="item.label"
|
||||
:value="item.value"
|
||||
:disabled="checkDisabled(item)"
|
||||
/>
|
||||
</el-select>
|
||||
</template>
|
||||
@@ -78,7 +79,7 @@ export default {
|
||||
},
|
||||
// 初始化值,也就是选中的值
|
||||
value: {
|
||||
type: [Array, String, Number, Boolean],
|
||||
type: [Array, String, Number, Boolean, Object],
|
||||
default() {
|
||||
return this.multiple ? [] : ''
|
||||
}
|
||||
@@ -90,6 +91,10 @@ export default {
|
||||
disabled: {
|
||||
type: Boolean,
|
||||
default: false
|
||||
},
|
||||
collapseTagsCount: {
|
||||
type: Number,
|
||||
default: 10
|
||||
}
|
||||
},
|
||||
data() {
|
||||
@@ -128,6 +133,9 @@ export default {
|
||||
selectRef() {
|
||||
return this.$refs.select
|
||||
},
|
||||
collapseTags() {
|
||||
return this.multiple && this.collapseTagsCount > 0 && this.value.length > this.collapseTagsCount
|
||||
},
|
||||
optionsValues() {
|
||||
return this.iOptions.map((v) => v.value)
|
||||
},
|
||||
@@ -137,7 +145,13 @@ export default {
|
||||
if (noValue && !this.initialized) {
|
||||
return
|
||||
}
|
||||
this.$emit('input', val)
|
||||
if (val && val.constructor === Object && val.value) {
|
||||
this.$emit('input', val.value)
|
||||
} else if (val && val.constructor === Object && val.id) {
|
||||
this.$emit('input', val.id)
|
||||
} else {
|
||||
this.$emit('input', val)
|
||||
}
|
||||
},
|
||||
get() {
|
||||
return this.value
|
||||
@@ -161,6 +175,10 @@ export default {
|
||||
return { label: item.name, value: item.id }
|
||||
}
|
||||
const transformOption = this.ajax.transformOption || defaultTransformOption
|
||||
const defaultFilterOption = (item) => {
|
||||
return item
|
||||
}
|
||||
const filterOption = this.ajax.filterOption || defaultFilterOption
|
||||
const defaultProcessResults = (data) => {
|
||||
let results = []
|
||||
let more = false
|
||||
@@ -174,7 +192,7 @@ export default {
|
||||
total = data.count
|
||||
}
|
||||
results = results.map(transformOption)
|
||||
results = results.filter(Boolean)
|
||||
results = results.filter(filterOption)
|
||||
return { results: results, pagination: more, total: total }
|
||||
}
|
||||
const defaultAjax = {
|
||||
@@ -189,31 +207,28 @@ export default {
|
||||
}
|
||||
},
|
||||
watch: {
|
||||
// url(newValue, oldValue) {
|
||||
// this.$log.debug('Select url changed: ', oldValue, ' => ', newValue)
|
||||
// this.iAjax.url = newValue
|
||||
// this.refresh()
|
||||
// },
|
||||
iAjax(newValue, oldValue) {
|
||||
this.$log.debug('Select url changed: ', oldValue, ' => ', newValue)
|
||||
this.refresh()
|
||||
},
|
||||
value(iNew) {
|
||||
this.iValue = iNew
|
||||
value: {
|
||||
handler(newValue, oldValue) {
|
||||
},
|
||||
deep: true
|
||||
}
|
||||
},
|
||||
async mounted() {
|
||||
// this.$log.debug('Select2 url is: ', this.iAjax.url)
|
||||
if (!this.initialized) {
|
||||
await this.initialSelect()
|
||||
setTimeout(() => {
|
||||
this.$log.debug('Value is : ', this.value)
|
||||
this.iValue = this.value
|
||||
this.initialized = true
|
||||
})
|
||||
}, 100)
|
||||
}
|
||||
this.$nextTick(() => {
|
||||
// 因为elform存在问题,这个来清楚验证
|
||||
const elFormItem = this.$refs.select.elFormItem
|
||||
const elFormItem = this.$refs.select?.elFormItem
|
||||
if (elFormItem && elFormItem.clearValidate) {
|
||||
elFormItem.clearValidate()
|
||||
}
|
||||
@@ -308,8 +323,9 @@ export default {
|
||||
if (this.iOptions.length === 0) {
|
||||
this.remote = false
|
||||
}
|
||||
} else {
|
||||
this.remote = false
|
||||
}
|
||||
this.iValue = this.value
|
||||
},
|
||||
refresh() {
|
||||
this.resetParams()
|
||||
@@ -319,13 +335,11 @@ export default {
|
||||
addOption(option) {
|
||||
this.iOptions.push(option)
|
||||
},
|
||||
getOptionsByValues(values) {
|
||||
return this.iOptions.filter((v) => {
|
||||
return values.indexOf(v.value) !== -1
|
||||
})
|
||||
},
|
||||
getSelectedOptions() {
|
||||
const values = this.iValue
|
||||
let values = this.iValue
|
||||
if (!Array.isArray(values)) {
|
||||
values = [values]
|
||||
}
|
||||
return this.iOptions.filter((v) => {
|
||||
return values.indexOf(v.value) !== -1
|
||||
})
|
||||
@@ -338,9 +352,9 @@ export default {
|
||||
},
|
||||
onChange(values) {
|
||||
const options = this.getSelectedOptions()
|
||||
this.$log.debug('Current select options: ', options)
|
||||
this.$log.debug('Current select options: ', options, 'Val: ', this.value)
|
||||
this.$emit('changeOptions', options)
|
||||
this.$emit('change', options)
|
||||
// this.$emit('change', options) // 事件重复
|
||||
},
|
||||
onVisibleChange(visible) {
|
||||
if (!visible && this.params.search) {
|
||||
@@ -354,11 +368,12 @@ export default {
|
||||
</script>
|
||||
|
||||
<style scoped>
|
||||
.select2 {
|
||||
width: 100%;
|
||||
}
|
||||
.select2 >>> .el-tag.el-tag--info {
|
||||
height: auto;
|
||||
white-space: normal;
|
||||
}
|
||||
.select2 {
|
||||
width: 100%;
|
||||
}
|
||||
|
||||
.select2 >>> .el-tag.el-tag--info {
|
||||
height: auto;
|
||||
white-space: normal;
|
||||
}
|
||||
</style>
|
||||
|
||||
@@ -1,8 +1,8 @@
|
||||
<template>
|
||||
<el-switch
|
||||
v-model="iValue"
|
||||
inactive-color="#dcdfe6"
|
||||
:class="type"
|
||||
inactive-color="#dcdfe6"
|
||||
v-bind="$attrs"
|
||||
v-on="$listeners"
|
||||
/>
|
||||
@@ -10,14 +10,14 @@
|
||||
|
||||
<script>
|
||||
export default {
|
||||
name: 'Switcher',
|
||||
name: 'Switcher', // 不能叫 Switch,否则会和 js 关键字冲突
|
||||
props: {
|
||||
type: {
|
||||
type: String,
|
||||
default: 'primary'
|
||||
},
|
||||
value: {
|
||||
type: Boolean,
|
||||
type: [Boolean, String],
|
||||
default: true
|
||||
}
|
||||
},
|
||||
@@ -31,9 +31,14 @@ export default {
|
||||
this.$emit('input', newValue)
|
||||
},
|
||||
get: function() {
|
||||
return this.value
|
||||
return !!this.value
|
||||
}
|
||||
}
|
||||
},
|
||||
watch: {
|
||||
value(val) {
|
||||
this.$log.debug('Switcher Value changed: ', val)
|
||||
}
|
||||
}
|
||||
}
|
||||
</script>
|
||||
143
src/components/FormFields/TagInput.vue
Normal file
143
src/components/FormFields/TagInput.vue
Normal file
@@ -0,0 +1,143 @@
|
||||
<template>
|
||||
<div class="filter-field">
|
||||
<el-tag
|
||||
v-for="(v, k) in filterTags"
|
||||
:key="k"
|
||||
:disable-transitions="true"
|
||||
:type="tagType(v)"
|
||||
closable
|
||||
size="small"
|
||||
@click="handleTagClick(v, k)"
|
||||
@close="handleTagClose(v)"
|
||||
>
|
||||
{{ v }}
|
||||
</el-tag>
|
||||
<component
|
||||
:is="component"
|
||||
ref="SearchInput"
|
||||
v-model.trim="filterValue"
|
||||
:fetch-suggestions="autocomplete"
|
||||
:placeholder="this.$t('common.EnterToContinue')"
|
||||
class="search-input"
|
||||
@blur="focus = false"
|
||||
@change="handleConfirm"
|
||||
@focus="focus = true"
|
||||
@select="handleSelect"
|
||||
@keyup.enter.native="handleConfirm"
|
||||
/>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<script>
|
||||
import i18n from '@/i18n/i18n'
|
||||
|
||||
export default {
|
||||
props: {
|
||||
value: {
|
||||
type: Array,
|
||||
default: () => []
|
||||
},
|
||||
tagType: {
|
||||
type: Function,
|
||||
default: () => {
|
||||
return 'info'
|
||||
}
|
||||
},
|
||||
placeholder: {
|
||||
type: String,
|
||||
default: () => i18n.t('perms.Input')
|
||||
},
|
||||
autocomplete: {
|
||||
type: Function,
|
||||
default: null
|
||||
}
|
||||
},
|
||||
data() {
|
||||
return {
|
||||
filterTags: this.value,
|
||||
focus: false,
|
||||
filterValue: '',
|
||||
component: this.autocomplete ? 'el-autocomplete' : 'el-input'
|
||||
}
|
||||
},
|
||||
watch: {
|
||||
value(val) {
|
||||
this.filterTags = val
|
||||
}
|
||||
},
|
||||
methods: {
|
||||
handleTagClose(tag) {
|
||||
this.filterTags.splice(this.filterTags.indexOf(tag), 1)
|
||||
this.$emit('change', this.filterTags)
|
||||
},
|
||||
handleSelect(item) {
|
||||
this.filterValue = item.value
|
||||
this.handleConfirm()
|
||||
},
|
||||
handleConfirm() {
|
||||
if (this.filterValue === '') return
|
||||
if (!this.filterTags.includes(this.filterValue)) {
|
||||
this.filterTags.push(this.filterValue)
|
||||
this.filterValue = ''
|
||||
this.$emit('change', this.filterTags)
|
||||
}
|
||||
},
|
||||
handleTagClick(v, k) {
|
||||
if (this.filterValue.length !== 0) {
|
||||
this.handleConfirm()
|
||||
}
|
||||
this.$delete(this.filterTags, k)
|
||||
this.filterValue = v
|
||||
this.$refs.SearchInput.focus()
|
||||
}
|
||||
}
|
||||
}
|
||||
</script>
|
||||
|
||||
<style lang="scss" scoped>
|
||||
.el-tag + .el-tag {
|
||||
margin-left: 4px;
|
||||
}
|
||||
|
||||
.filter-field {
|
||||
display: flex;
|
||||
flex-wrap: wrap;
|
||||
align-items: center;
|
||||
padding-left: 2px;
|
||||
padding-bottom: 3px;
|
||||
border: 1px solid #dcdee2;
|
||||
border-radius: 1px;
|
||||
background-color: #fff;
|
||||
line-height: 32px;
|
||||
|
||||
&:hover {
|
||||
border-color: #C0C4CC;
|
||||
}
|
||||
|
||||
&>>> .el-tag {
|
||||
margin-top: 3px;
|
||||
}
|
||||
|
||||
&>>> .el-autocomplete {
|
||||
height: 26px;
|
||||
}
|
||||
}
|
||||
|
||||
.search-input {
|
||||
flex: 1;
|
||||
&>>> .el-input__inner {
|
||||
max-width: 100%;
|
||||
border: none;
|
||||
padding-left: 5px;
|
||||
}
|
||||
}
|
||||
|
||||
.el-input >>> .el-input__inner {
|
||||
border: none !important;
|
||||
font-size: 13px;
|
||||
}
|
||||
|
||||
.filter-field >>> .el-input__inner {
|
||||
height: 26px;
|
||||
}
|
||||
</style>
|
||||
36
src/components/FormFields/TextReadonly.vue
Normal file
36
src/components/FormFields/TextReadonly.vue
Normal file
@@ -0,0 +1,36 @@
|
||||
<template>
|
||||
<div>
|
||||
{{ value? trueText : falseText }}
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<script>
|
||||
export default {
|
||||
props: {
|
||||
value: {
|
||||
type: [String, Boolean],
|
||||
default: () => false
|
||||
},
|
||||
trueText: {
|
||||
type: String,
|
||||
default: function() {
|
||||
return this.$t('common.Yes')
|
||||
}
|
||||
},
|
||||
falseText: {
|
||||
type: String,
|
||||
default: function() {
|
||||
return this.$t('common.No')
|
||||
}
|
||||
}
|
||||
|
||||
},
|
||||
data() {
|
||||
return {}
|
||||
}
|
||||
}
|
||||
</script>
|
||||
|
||||
<style scoped>
|
||||
|
||||
</style>
|
||||
92
src/components/FormFields/UpdateSelect.vue
Normal file
92
src/components/FormFields/UpdateSelect.vue
Normal file
@@ -0,0 +1,92 @@
|
||||
<template>
|
||||
<div>
|
||||
<el-button
|
||||
v-show="!iShowSelect"
|
||||
:disabled="disabled"
|
||||
class="button-text"
|
||||
type="text"
|
||||
@click="iShowSelect=true"
|
||||
>
|
||||
{{ iLabel }}
|
||||
<svg-icon class-name="icon" icon-class="switch" />
|
||||
</el-button>
|
||||
<Select2
|
||||
v-show="iShowSelect"
|
||||
ref="select2"
|
||||
v-model="iValue"
|
||||
:disabled="disabled"
|
||||
v-bind="$attrs"
|
||||
@change="onSelectChange"
|
||||
v-on="$listeners"
|
||||
/>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<script>
|
||||
import Select2 from './Select2'
|
||||
import { hasUUID } from '@/utils/common'
|
||||
|
||||
export default {
|
||||
components: {
|
||||
Select2
|
||||
},
|
||||
props: {
|
||||
value: {
|
||||
type: String,
|
||||
default: () => ''
|
||||
},
|
||||
label: {
|
||||
type: String,
|
||||
default: () => ''
|
||||
},
|
||||
showSelect: {
|
||||
type: Boolean,
|
||||
default: false
|
||||
},
|
||||
disabled: {
|
||||
type: Boolean,
|
||||
default: false
|
||||
}
|
||||
},
|
||||
data() {
|
||||
return {
|
||||
iShowSelect: this.showSelect,
|
||||
iLabel: this.label || '-'
|
||||
}
|
||||
},
|
||||
computed: {
|
||||
iValue: {
|
||||
get() {
|
||||
return this.value
|
||||
},
|
||||
set(val) {
|
||||
this.$emit('input', val)
|
||||
}
|
||||
}
|
||||
},
|
||||
created() {
|
||||
const { path } = this.$route
|
||||
if (hasUUID(path) && this.value) {
|
||||
this.iShowSelect = false
|
||||
}
|
||||
},
|
||||
methods: {
|
||||
onSelectChange(val) {
|
||||
const options = this.$refs.select2.options.filter(item => item.value === val)
|
||||
const label = options.length > 0 ? options[0].label : ''
|
||||
this.iShowSelect = false
|
||||
this.iLabel = val ? label : '-'
|
||||
}
|
||||
}
|
||||
}
|
||||
</script>
|
||||
|
||||
<style scoped>
|
||||
.button-text {
|
||||
color: #676a6c;
|
||||
padding: 5px!important;
|
||||
}
|
||||
.icon {
|
||||
color: #676a6c!important;
|
||||
}
|
||||
</style>
|
||||
@@ -32,6 +32,10 @@ export default {
|
||||
return this.$t('common.Update')
|
||||
}
|
||||
},
|
||||
showInput: {
|
||||
type: Boolean,
|
||||
default: true
|
||||
},
|
||||
placeholder: {
|
||||
type: String,
|
||||
default: () => ''
|
||||
@@ -39,13 +43,13 @@ export default {
|
||||
},
|
||||
data() {
|
||||
return {
|
||||
isShow: false,
|
||||
isShow: this.showInput,
|
||||
curValue: this.value
|
||||
}
|
||||
},
|
||||
created() {
|
||||
if (this.$route.path.indexOf('/create') !== -1) {
|
||||
this.isShow = true
|
||||
if (this.$route.path.indexOf('/update') !== -1) {
|
||||
this.isShow = false
|
||||
}
|
||||
},
|
||||
methods: {
|
||||
|
||||
@@ -1,6 +1,6 @@
|
||||
<template>
|
||||
<div>
|
||||
<input ref="upLoadFile" type="file" style="display: none" @change="Onchange">
|
||||
<input ref="upLoadFile" :accept="accept" type="file" style="display: none" @change="Onchange">
|
||||
<el-button size="mini" @click.native.stop="onUpLoad">
|
||||
{{ this.$t('common.SelectFile') }}
|
||||
</el-button>
|
||||
@@ -23,6 +23,10 @@ export default {
|
||||
tip: {
|
||||
type: String,
|
||||
default: () => ''
|
||||
},
|
||||
accept: {
|
||||
type: String,
|
||||
default: '*'
|
||||
}
|
||||
},
|
||||
data() {
|
||||
|
||||
115
src/components/FormFields/UploadSecret.vue
Normal file
115
src/components/FormFields/UploadSecret.vue
Normal file
@@ -0,0 +1,115 @@
|
||||
<template>
|
||||
<div class="">
|
||||
<el-input
|
||||
v-model="iValue"
|
||||
type="textarea"
|
||||
:rows="rows"
|
||||
:placeholder="placeholder"
|
||||
/>
|
||||
<el-upload
|
||||
ref="upload"
|
||||
class="upload-secret"
|
||||
:action="''"
|
||||
:accept="accept"
|
||||
:auto-upload="false"
|
||||
:limit="limit"
|
||||
v-bind="$attrs"
|
||||
:on-change="handleChange"
|
||||
:on-remove="handleRemove"
|
||||
:file-list="fileList"
|
||||
>
|
||||
<el-button size="mini" type="primary">
|
||||
{{ btnText }}
|
||||
</el-button>
|
||||
<div v-if="tip" slot="tip" class="el-upload__tip">
|
||||
{{ tip }}
|
||||
</div>
|
||||
</el-upload>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<script>
|
||||
export default {
|
||||
props: {
|
||||
value: {
|
||||
type: String,
|
||||
default: () => ''
|
||||
},
|
||||
btnText: {
|
||||
type: String,
|
||||
default: function() {
|
||||
return this.$t('common.SelectFile')
|
||||
}
|
||||
},
|
||||
rows: {
|
||||
type: Number,
|
||||
default: () => 4
|
||||
},
|
||||
limit: {
|
||||
type: Number,
|
||||
default: () => 2
|
||||
},
|
||||
accept: {
|
||||
type: String,
|
||||
default: () => ''
|
||||
},
|
||||
placeholder: {
|
||||
type: String,
|
||||
default: () => ''
|
||||
},
|
||||
tip: {
|
||||
type: String,
|
||||
default: () => ''
|
||||
}
|
||||
},
|
||||
data() {
|
||||
return {
|
||||
fileName: '',
|
||||
fileList: [],
|
||||
iValue: this.value
|
||||
}
|
||||
},
|
||||
watch: {
|
||||
iValue(val) {
|
||||
this.$emit('input', val)
|
||||
}
|
||||
},
|
||||
methods: {
|
||||
handleChange(file, fileList) {
|
||||
const vm = this
|
||||
const newFileList = fileList.slice(-1)
|
||||
this.fileList = newFileList
|
||||
const reader = new FileReader()
|
||||
reader.onload = function(res) {
|
||||
const result = res.target.result
|
||||
vm.iValue = result
|
||||
vm.$emit('input', vm.iValue)
|
||||
}
|
||||
reader.readAsText(file.raw)
|
||||
},
|
||||
|
||||
handleRemove() {
|
||||
this.iValue = ''
|
||||
this.fileList = []
|
||||
this.$emit('input', this.iValue)
|
||||
}
|
||||
}
|
||||
}
|
||||
</script>
|
||||
|
||||
<style lang="scss" scoped>
|
||||
.upload-secret {
|
||||
display: flex;
|
||||
&>>> .el-list-enter-active,
|
||||
&>>> .el-list-leave-active {
|
||||
transition: none;
|
||||
}
|
||||
&>>> .el-list-enter,
|
||||
&>>> .el-list-leave-active {
|
||||
opacity: 0;
|
||||
}
|
||||
&>>> .el-upload-list {
|
||||
height: 40px;
|
||||
}
|
||||
}
|
||||
</style>
|
||||
@@ -2,10 +2,10 @@
|
||||
<div class="c-weektime">
|
||||
<div class="c-schedue" />
|
||||
<div :class="{'c-schedue': true, 'c-schedue-notransi': mode}" :style="styleValue" />
|
||||
<table class="c-weektime-table" :class="{'c-min-table': colspan < 2}">
|
||||
<table :class="{'c-min-table': colspan < 2}" class="c-weektime-table">
|
||||
<thead class="c-weektime-head">
|
||||
<tr>
|
||||
<th rowspan="8" class="week-td">{{ this.$t('common.WeekCronSelect.WeekOrTime') }}</th>
|
||||
<th class="week-td" rowspan="8">{{ this.$t('common.WeekCronSelect.WeekOrTime') }}</th>
|
||||
<th :colspan="12 * colspan">00:00 - 12:00</th>
|
||||
<th :colspan="12 * colspan">12:00 - 24:00</th>
|
||||
</tr>
|
||||
@@ -13,23 +13,23 @@
|
||||
<td v-for="t in theadArr" :key="t" :colspan="colspan">{{ t }}</td>
|
||||
</tr>
|
||||
</thead>
|
||||
<tbody class="c-weektime-body">
|
||||
<tbody class="c-weektime-body" @mouseleave="containerLeave()">
|
||||
<tr v-for="t in weektimeData" :key="t.row">
|
||||
<td>{{ t.value }}</td>
|
||||
<td
|
||||
v-for="n in t.child"
|
||||
:key="`${n.row}-${n.col}`"
|
||||
:data-week="n.row"
|
||||
:data-time="n.col"
|
||||
:class="selectClasses(n)"
|
||||
:data-time="n.col"
|
||||
:data-week="n.row"
|
||||
class="weektime-atom-item"
|
||||
@mouseenter="cellEnter(n)"
|
||||
@mousedown="cellDown(n)"
|
||||
@mouseenter="cellEnter(n)"
|
||||
@mouseup="cellUp(n)"
|
||||
/>
|
||||
</tr>
|
||||
<tr>
|
||||
<td colspan="49" class="c-weektime-preview">
|
||||
<td class="c-weektime-preview" colspan="49">
|
||||
<div class="g-clearfix c-weektime-con">
|
||||
<span class="g-pull-left">{{ this.$t('common.WeekCronSelect.CanDragSelect') }}</span>
|
||||
<a class="g-pull-right" @click.prevent="clearWeektime">{{ this.$t('common.WeekCronSelect.ClearSelection') }}</a>
|
||||
@@ -230,6 +230,11 @@ export default {
|
||||
})
|
||||
this.setTimeRange()
|
||||
},
|
||||
containerLeave() {
|
||||
this.width = 0
|
||||
this.height = 0
|
||||
this.mode = 0
|
||||
},
|
||||
setTimeRange() {
|
||||
this.timeRange = this.weektimeData.map(item => {
|
||||
return {
|
||||
@@ -314,6 +319,7 @@ export default {
|
||||
min-width: 640px;
|
||||
position: relative;
|
||||
display: inline-block;
|
||||
padding-right: 20px;
|
||||
}
|
||||
.c-schedue {
|
||||
background: #598fe6;
|
||||
|
||||
@@ -1,42 +1,60 @@
|
||||
import DatetimeRangePicker from './DatetimeRangePicker'
|
||||
import Link from './Link'
|
||||
import PasswordInput from './PasswordInput'
|
||||
import Select2 from './Select2'
|
||||
import Swicher from './Swicher'
|
||||
import UploadField from './UploadField'
|
||||
import UploadKey from './UploadKey'
|
||||
import UserPassword from './UserPassword'
|
||||
import WeekCronSelect from './WeekCronSelect'
|
||||
import UpdateToken from './UpdateToken'
|
||||
import JsonEditor from './JsonEditor'
|
||||
import Text from './Text'
|
||||
import Select2 from './Select2'
|
||||
import TagInput from './TagInput'
|
||||
import Switcher from './Switcher'
|
||||
import UploadKey from './UploadKey'
|
||||
import JsonEditor from './JsonEditor'
|
||||
import PhoneInput from './PhoneInput'
|
||||
import UploadField from './UploadField'
|
||||
import UpdateToken from './UpdateToken'
|
||||
import UserPassword from './UserPassword'
|
||||
import DynamicInput from './DynamicInput'
|
||||
import PasswordInput from './PasswordInput'
|
||||
import UploadSecret from './UploadSecret'
|
||||
import WeekCronSelect from './WeekCronSelect'
|
||||
import NestedObjectSelect2 from './NestedObjectSelect2'
|
||||
import DatetimeRangePicker from './DatetimeRangePicker'
|
||||
import JSONManyToManySelect from './JSONManyToManySelect/index.vue'
|
||||
|
||||
export default {
|
||||
DatetimeRangePicker,
|
||||
Text,
|
||||
Link,
|
||||
PasswordInput,
|
||||
Switcher,
|
||||
Select2,
|
||||
Swicher,
|
||||
TagInput,
|
||||
UploadKey,
|
||||
JsonEditor,
|
||||
UpdateToken,
|
||||
PhoneInput,
|
||||
UploadField,
|
||||
UserPassword,
|
||||
DynamicInput,
|
||||
PasswordInput,
|
||||
UploadSecret,
|
||||
WeekCronSelect,
|
||||
UpdateToken,
|
||||
JsonEditor,
|
||||
Text
|
||||
NestedObjectSelect2,
|
||||
DatetimeRangePicker,
|
||||
JSONManyToManySelect
|
||||
}
|
||||
|
||||
export {
|
||||
DatetimeRangePicker,
|
||||
Text,
|
||||
Link,
|
||||
PasswordInput,
|
||||
Switcher,
|
||||
Select2,
|
||||
Swicher,
|
||||
TagInput,
|
||||
UploadKey,
|
||||
JsonEditor,
|
||||
UpdateToken,
|
||||
PhoneInput,
|
||||
UploadField,
|
||||
UserPassword,
|
||||
DynamicInput,
|
||||
PasswordInput,
|
||||
UploadSecret,
|
||||
WeekCronSelect,
|
||||
UpdateToken,
|
||||
JsonEditor,
|
||||
Text
|
||||
NestedObjectSelect2,
|
||||
DatetimeRangePicker,
|
||||
JSONManyToManySelect
|
||||
}
|
||||
|
||||
@@ -1,7 +1,7 @@
|
||||
<template>
|
||||
<div class="form-group-header">
|
||||
<div v-if="line" class="hr-line-dashed" />
|
||||
<h3>{{ group.title }}</h3>
|
||||
<h3>{{ group['title'] }} </h3>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
|
||||
92
src/components/GatewayDialog/index.vue
Normal file
92
src/components/GatewayDialog/index.vue
Normal file
@@ -0,0 +1,92 @@
|
||||
<template>
|
||||
<Dialog
|
||||
v-if="iVisible"
|
||||
:destroy-on-close="true"
|
||||
:show-cancel="false"
|
||||
:show-confirm="false"
|
||||
:title="$tc('assets.TestGatewayTestConnection')"
|
||||
:visible.sync="iVisible"
|
||||
top="35vh"
|
||||
width="40%"
|
||||
>
|
||||
<el-row :gutter="20">
|
||||
<el-col :md="4" :sm="24">
|
||||
<div style="line-height: 34px">{{ $t('assets.SSHPort') }}</div>
|
||||
</el-col>
|
||||
<el-col :md="14" :sm="24">
|
||||
<el-input v-model="port" />
|
||||
<span class="help-tips help-block">{{ $t('assets.TestGatewayHelpMessage') }}</span>
|
||||
</el-col>
|
||||
<el-col :md="4" :sm="24">
|
||||
<el-button
|
||||
:loading="loading"
|
||||
size="mini"
|
||||
style="line-height:20px "
|
||||
type="primary"
|
||||
@click="dialogConfirm"
|
||||
>
|
||||
{{ this.$t('common.Confirm') }}
|
||||
</el-button>
|
||||
</el-col>
|
||||
</el-row>
|
||||
</Dialog>
|
||||
</template>
|
||||
|
||||
<script>
|
||||
import Dialog from '@/components/Dialog'
|
||||
import { openTaskPage } from '@/utils/jms'
|
||||
|
||||
export default {
|
||||
name: 'GatewayDialog',
|
||||
components: {
|
||||
Dialog
|
||||
},
|
||||
props: {
|
||||
visible: {
|
||||
type: Boolean,
|
||||
default: false
|
||||
},
|
||||
loading: {
|
||||
type: Boolean,
|
||||
default: false
|
||||
},
|
||||
port: {
|
||||
type: Number,
|
||||
default: 0
|
||||
},
|
||||
cell: {
|
||||
type: String,
|
||||
default: ''
|
||||
}
|
||||
},
|
||||
data() {
|
||||
return {}
|
||||
},
|
||||
computed: {
|
||||
iVisible: {
|
||||
get() {
|
||||
return this.visible
|
||||
},
|
||||
set(val) {
|
||||
this.$emit('update:visible', val)
|
||||
}
|
||||
}
|
||||
},
|
||||
methods: {
|
||||
dialogConfirm() {
|
||||
if (isNaN(this.port)) {
|
||||
return this.$message.error(this.$tc('common.TestPortErrorMsg'))
|
||||
}
|
||||
this.$axios.post(
|
||||
`/api/v1/assets/gateways/${this.cell}/test-connective/`,
|
||||
{ port: this.port }
|
||||
)
|
||||
.then((res) => {
|
||||
openTaskPage(res['task'])
|
||||
}).finally(() => {
|
||||
this.iVisible = false
|
||||
})
|
||||
}
|
||||
}
|
||||
}
|
||||
</script>
|
||||
@@ -1,10 +1,10 @@
|
||||
<template>
|
||||
<TreeTable :table-config="tableConfig" :header-actions="headerActions" :tree-setting="treeSetting" />
|
||||
<TreeTable :header-actions="headerActions" :table-config="tableConfig" :tree-setting="treeSetting" />
|
||||
</template>
|
||||
|
||||
<script type="text/jsx">
|
||||
import { DetailFormatter, SystemUserFormatter } from '@/components/TableFormatters'
|
||||
import TreeTable from '../TreeTable'
|
||||
import { DetailFormatter } from '@/components/TableFormatters'
|
||||
|
||||
export default {
|
||||
name: 'GrantedAssets',
|
||||
@@ -35,7 +35,7 @@ export default {
|
||||
getShowUrl: {
|
||||
type: Function,
|
||||
default({ row, col }) {
|
||||
return this.tableUrl.replace('/assets/', `/assets/${row.id}/system-users/?cache_policy=1`)
|
||||
return this.tableUrl.replace('/assets/', `/assets/${row.id}/accounts/`)
|
||||
}
|
||||
}
|
||||
},
|
||||
@@ -57,34 +57,22 @@ export default {
|
||||
tableConfig: {
|
||||
url: this.tableUrl,
|
||||
hasTree: true,
|
||||
columns: [
|
||||
{
|
||||
prop: 'hostname',
|
||||
label: this.$t('assets.Hostname'),
|
||||
columnsExclude: ['spec_info'],
|
||||
columnShow: {
|
||||
min: ['name', 'address', 'accounts'],
|
||||
default: ['name', 'address', 'accounts', 'actions']
|
||||
},
|
||||
columnsMeta: {
|
||||
name: {
|
||||
formatter: DetailFormatter,
|
||||
sortable: true,
|
||||
formatterArgs: {
|
||||
route: 'AssetDetail'
|
||||
},
|
||||
showOverflowTooltip: true
|
||||
}
|
||||
},
|
||||
{
|
||||
prop: 'ip',
|
||||
label: this.$t('assets.IP'),
|
||||
width: '140px',
|
||||
sortable: 'custom'
|
||||
},
|
||||
{
|
||||
prop: 'systemUsers',
|
||||
label: this.$t('assets.SystemUsers'),
|
||||
align: 'center',
|
||||
formatter: SystemUserFormatter,
|
||||
formatterArgs: {
|
||||
getUrl: this.getShowUrl.bind(this)
|
||||
},
|
||||
showOverflowTooltip: true
|
||||
actions: {
|
||||
has: false
|
||||
}
|
||||
]
|
||||
}
|
||||
},
|
||||
headerActions: {
|
||||
hasLeftActions: false,
|
||||
|
||||
@@ -1,6 +1,6 @@
|
||||
<template>
|
||||
<div style="padding: 0 15px;" @click="toggleClick">
|
||||
<svg-icon icon-class="hamburger" class="hamburger" :class="{'is-active':isActive}" />
|
||||
<div style="padding: 0 20px;" @click="toggleClick">
|
||||
<svg-icon icon-class="arrow-to-left" class="hamburger" style="color: #ffffff;" :class="{'is-active':isActive}" />
|
||||
</div>
|
||||
</template>
|
||||
|
||||
@@ -26,7 +26,7 @@ export default {
|
||||
.hamburger {
|
||||
display: inline-block;
|
||||
vertical-align: middle;
|
||||
font-size: 20px;
|
||||
font-size: 16px;
|
||||
color: $menuText;
|
||||
}
|
||||
.hamburger.is-active {
|
||||
|
||||
@@ -3,7 +3,7 @@
|
||||
<template #header>
|
||||
<slot name="header">
|
||||
<div v-if="title" slot="header" class="clearfix ibox-title">
|
||||
<i v-if="fa" :class="'fa ' + fa" /> {{ title }}
|
||||
<i v-if="fa" :class="'fa ' + fa" /> <h5>{{ title }}</h5>
|
||||
</div>
|
||||
</slot>
|
||||
</template>
|
||||
@@ -41,7 +41,6 @@ export default {
|
||||
/*height: 100%;*/
|
||||
clear: both;
|
||||
padding: 0;
|
||||
background-color: #ffffff;
|
||||
}
|
||||
|
||||
.ibox >>> .el-card__header {
|
||||
@@ -56,11 +55,11 @@ export default {
|
||||
|
||||
.ibox-title h5 {
|
||||
display: inline-block;
|
||||
font-size: 14px;
|
||||
font-size: 13px;
|
||||
margin: 0;
|
||||
padding: 0;
|
||||
text-overflow: ellipsis;
|
||||
float: left;
|
||||
font-weight: 500;
|
||||
}
|
||||
|
||||
.ibox-tools a {
|
||||
@@ -83,8 +82,7 @@ export default {
|
||||
}
|
||||
|
||||
.ibox >>> .el-card__body {
|
||||
background-color: #ffffff;
|
||||
padding: 15px 20px 20px 20px;
|
||||
padding: 15px 30px 20px 30px;
|
||||
color: inherit;
|
||||
}
|
||||
</style>
|
||||
|
||||
36
src/components/Link/index.vue
Normal file
36
src/components/Link/index.vue
Normal file
@@ -0,0 +1,36 @@
|
||||
<template>
|
||||
<el-link v-right-click-target="target" :href="url" @click.prevent="openLink">
|
||||
<slot />
|
||||
</el-link>
|
||||
</template>
|
||||
|
||||
<script>
|
||||
export default {
|
||||
name: 'RightClickLink',
|
||||
directives: {
|
||||
'right-click-target': {
|
||||
inserted(el, binding) {
|
||||
el.addEventListener('contextmenu', (event) => {
|
||||
event.preventDefault()
|
||||
window.open(binding.value, '_blank')
|
||||
})
|
||||
}
|
||||
}
|
||||
},
|
||||
props: {
|
||||
url: {
|
||||
type: String,
|
||||
required: true
|
||||
},
|
||||
target: {
|
||||
type: String,
|
||||
default: '_blank'
|
||||
}
|
||||
},
|
||||
methods: {
|
||||
openLink() {
|
||||
window.open(this.url, this.target)
|
||||
}
|
||||
}
|
||||
}
|
||||
</script>
|
||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user