mirror of
https://github.com/chylex/IntelliJ-IdeaVim.git
synced 2025-10-24 09:23:43 +02:00
Compare commits
564 Commits
customized
...
54c16c97b9
Author | SHA1 | Date | |
---|---|---|---|
54c16c97b9
|
|||
![]() |
5b9cb2efc5 | ||
![]() |
733968723c | ||
![]() |
63c81d67f2 | ||
![]() |
ad8ba1dd24 | ||
![]() |
04f821e3e1 | ||
![]() |
4937985e2c | ||
![]() |
5fd7d83a70 | ||
![]() |
699a19d202 | ||
![]() |
0b42938197 | ||
![]() |
1e2bfb6216 | ||
![]() |
f755a4b23f | ||
![]() |
4f58e12fca | ||
![]() |
588cf89531 | ||
![]() |
4fe2dd2706 | ||
![]() |
11ad605cd6 | ||
![]() |
fa9f160bd1 | ||
![]() |
dae1fad54e | ||
![]() |
52200188d4 | ||
![]() |
0d74b9ef0b | ||
![]() |
549163d274 | ||
![]() |
755018c783 | ||
![]() |
2a1c4b3a1c | ||
![]() |
aae0d825e7 | ||
![]() |
855dbfab16 | ||
![]() |
f3a357c559 | ||
![]() |
63995e8c61 | ||
![]() |
7062d9b8f8 | ||
![]() |
ede62f5c75 | ||
![]() |
6386770ff3 | ||
![]() |
b4e831a81f | ||
![]() |
9b283360ce | ||
![]() |
fabbd4d156 | ||
![]() |
9bea5bf5f7 | ||
![]() |
9fbc990493 | ||
![]() |
b05fdaaa73 | ||
![]() |
52d5d4d64c | ||
![]() |
6ec712466c | ||
![]() |
6616b8dc07 | ||
![]() |
807457c718 | ||
![]() |
13d2a40903 | ||
![]() |
022b196d6a | ||
![]() |
7a64216830 | ||
![]() |
bf7d2bd465 | ||
![]() |
6e97b591de | ||
![]() |
fc7c470966 | ||
![]() |
51492ca121 | ||
![]() |
ce1df84330 | ||
![]() |
9b43e2a715 | ||
![]() |
732cabd6aa | ||
![]() |
7c14801d5c | ||
![]() |
66df09c065 | ||
![]() |
8fd6985316 | ||
![]() |
feac001499 | ||
![]() |
4c47e3a8eb | ||
![]() |
7773b625a5 | ||
![]() |
abe1abec72 | ||
![]() |
023838a96b | ||
![]() |
f4e743acc5 | ||
![]() |
06d58cbda5 | ||
![]() |
d199dea204 | ||
![]() |
5722060ed9 | ||
![]() |
d4f7e727c1 | ||
![]() |
ba9afc3f8e | ||
![]() |
39897bd012 | ||
![]() |
575d563154 | ||
![]() |
2bf46ce2f3 | ||
![]() |
b49a185efc | ||
![]() |
e305ebd1ed | ||
![]() |
6f5c9826f4 | ||
![]() |
6025eaaca9 | ||
![]() |
b2441c3cca | ||
![]() |
a73599e9ee | ||
![]() |
58398f40fa | ||
![]() |
43f5d5a8e8 | ||
![]() |
b20cbd3558 | ||
![]() |
7f835a407c | ||
![]() |
9859974db7 | ||
![]() |
6c24ddd1a0 | ||
![]() |
bd92ef08ec | ||
![]() |
8de6107a17 | ||
![]() |
e639f03ac7 | ||
![]() |
f9aac442c1 | ||
![]() |
5fdf675168 | ||
![]() |
232f81ff48 | ||
![]() |
1c4a6b2274 | ||
![]() |
deb71f8efc | ||
![]() |
4596596d9f | ||
![]() |
bbb6d42f8d | ||
![]() |
01efd0f9f0 | ||
![]() |
2d7597d206 | ||
![]() |
221741c891 | ||
![]() |
9f69beb450 | ||
![]() |
e843d9e9c3 | ||
![]() |
008b3d94fb | ||
![]() |
6756d83c55 | ||
![]() |
b52072a2e3 | ||
![]() |
3afb00d563 | ||
![]() |
a30c94fd2f | ||
![]() |
f50c29a285 | ||
![]() |
f238b0f138 | ||
![]() |
d0a8c98040 | ||
![]() |
b3d161ad97 | ||
![]() |
fce9cf2077 | ||
![]() |
efd0e56697 | ||
![]() |
b94a9bb9d9 | ||
![]() |
c153cc5a29 | ||
![]() |
a680e9a25a | ||
![]() |
3c18c4ef22 | ||
![]() |
c4e11b5976 | ||
![]() |
65be51dd48 | ||
![]() |
9684103f97 | ||
![]() |
f4c647d430 | ||
![]() |
f1eab3b9c1 | ||
![]() |
545d52bd93 | ||
![]() |
4e42198c09 | ||
![]() |
44736a51b9 | ||
![]() |
e675ffd623 | ||
![]() |
1f14e06bd3 | ||
![]() |
9871078269 | ||
![]() |
5e7a7f4d62 | ||
![]() |
7d690c6809 | ||
![]() |
6edb4266d5 | ||
![]() |
799e82d501 | ||
![]() |
a2370bff68 | ||
![]() |
c72f3bcd12 | ||
![]() |
295964a74d | ||
![]() |
d77cda0fae | ||
![]() |
6da072d47d | ||
![]() |
471a5a1b3e | ||
![]() |
cd5da2d237 | ||
![]() |
62f67cd626 | ||
![]() |
70db96d9e5 | ||
![]() |
98470111fb | ||
![]() |
557a3bb01f | ||
![]() |
dee70acdcb | ||
![]() |
862b16879c | ||
![]() |
ed7249558e | ||
![]() |
4f6c6f4d10 | ||
![]() |
650d02d9b3 | ||
![]() |
e4041a2f69 | ||
![]() |
4c284a6d13 | ||
![]() |
e14fc801bd | ||
![]() |
0478d468e0 | ||
![]() |
4ac98710fb | ||
![]() |
f256f6417e | ||
![]() |
ca94d55b62 | ||
![]() |
c11c061113 | ||
![]() |
c15c3eb802 | ||
![]() |
0ce102b782 | ||
![]() |
cc48207a99 | ||
![]() |
353ea5fc5d | ||
![]() |
64138310cc | ||
![]() |
1c4538af72 | ||
![]() |
755b47ef19 | ||
![]() |
c78a5d3cab | ||
![]() |
b9b8d30f3b | ||
![]() |
9be93212c3 | ||
![]() |
89973809af | ||
![]() |
e324af356d | ||
![]() |
f51fc6ed47 | ||
![]() |
ecce98289a | ||
![]() |
23c14aa2e4 | ||
![]() |
678d04c5db | ||
![]() |
691ba75372 | ||
![]() |
d2d7bbc632 | ||
![]() |
b3b1a6bdb9 | ||
![]() |
310125ea01 | ||
![]() |
208d1cbba2 | ||
![]() |
e94154ba80 | ||
![]() |
582fbdd9e7 | ||
![]() |
dd175912f4 | ||
![]() |
a6a0ae7a51 | ||
![]() |
8cdac91a01 | ||
![]() |
4c89f41daa | ||
![]() |
512e826a42 | ||
![]() |
bc0d277a21 | ||
![]() |
169fe5fc5b | ||
![]() |
30867702a4 | ||
![]() |
6131f92ae6 | ||
![]() |
823a52583c | ||
![]() |
e2c6c0539f | ||
![]() |
f7f1c0e90d | ||
![]() |
eca12607dd | ||
![]() |
006e3e11f9 | ||
![]() |
a9982cbdca | ||
![]() |
0fa9c5a2a2 | ||
![]() |
cdcc9729d3 | ||
![]() |
4acf651aa7 | ||
![]() |
4bba791c65 | ||
![]() |
662688d3b9 | ||
![]() |
21a3e8fdc4 | ||
![]() |
3815a1d538 | ||
![]() |
cbe0c5cfec | ||
![]() |
15db9b30e1 | ||
![]() |
e891294c0f | ||
![]() |
f6b9e7cc26 | ||
![]() |
052fd7162f | ||
![]() |
189acb73f5 | ||
![]() |
ec7c1677b4 | ||
![]() |
a9474c8e67 | ||
![]() |
3a70dfc5f3 | ||
![]() |
669177d803 | ||
![]() |
b1f43b061c | ||
![]() |
7ff3c84deb | ||
![]() |
ee642b63ce | ||
![]() |
17315e5096 | ||
![]() |
4e9d52fc62 | ||
![]() |
d7e87f8fc8 | ||
![]() |
3efe11f393 | ||
![]() |
26c6c464d8 | ||
![]() |
4db654e653 | ||
![]() |
048759d374 | ||
![]() |
db2424057f | ||
![]() |
472a53e3b9 | ||
![]() |
9e15d91900 | ||
![]() |
d5cff281c0 | ||
![]() |
57b6c4dffb | ||
![]() |
908a2d1d8c | ||
![]() |
69bdea9273 | ||
![]() |
5b21a653ee | ||
![]() |
cfddcf1630 | ||
![]() |
f009687ddf | ||
![]() |
6ddfe29465 | ||
![]() |
715c51f673 | ||
![]() |
b443e8f06a | ||
![]() |
0bd0466c9c | ||
![]() |
ad5db3c9e5 | ||
![]() |
fa3182cb5e | ||
![]() |
3f44bed66e | ||
![]() |
2a70530d0f | ||
![]() |
7c542d5fc7 | ||
![]() |
638dfb7777 | ||
![]() |
1323536a63 | ||
![]() |
419212e2d4 | ||
![]() |
5f1c234a7d | ||
![]() |
db1e8301cd | ||
![]() |
bf94a3c68d | ||
![]() |
96baa4ffc6 | ||
![]() |
7d472afe61 | ||
![]() |
f32a4d33a7 | ||
![]() |
0722991955 | ||
![]() |
bcc740cdbc | ||
![]() |
5cf46097f7 | ||
![]() |
61dc189f8b | ||
![]() |
23c2b008c9 | ||
![]() |
db14afdf3a | ||
![]() |
b7927336d1 | ||
![]() |
ee23a3d4cd | ||
![]() |
63c0112ffb | ||
![]() |
db08d7d280 | ||
![]() |
9892525fbc | ||
![]() |
34b87ff6bf | ||
![]() |
241ad68bd5 | ||
![]() |
a0ec18921b | ||
![]() |
45e17eb0b2 | ||
![]() |
59f0e9ae67 | ||
![]() |
af24611c73 | ||
![]() |
d4502dda3f | ||
![]() |
c0efa8af5d | ||
![]() |
1c06a3fc89 | ||
![]() |
c19fb38d1c | ||
![]() |
5dc1de9daf | ||
![]() |
6774301938 | ||
![]() |
4ef6cf0428 | ||
![]() |
ca5f8e4b44 | ||
![]() |
1907f03abe | ||
![]() |
6351a4e4f3 | ||
![]() |
fa34c3937f | ||
![]() |
cdac97ebf5 | ||
![]() |
fe958d28b8 | ||
![]() |
f71982e1d5 | ||
![]() |
cb2bfcea53 | ||
![]() |
4a9d5bbceb | ||
![]() |
10809eade6 | ||
![]() |
43d63527f8 | ||
![]() |
df51eb54ed | ||
![]() |
b47109ab4d | ||
![]() |
15b2b68940 | ||
![]() |
62a239f6fe | ||
![]() |
d89bc95a0a | ||
![]() |
2a76f21b31 | ||
![]() |
f07e22d742 | ||
![]() |
058ab7a1ea | ||
![]() |
fae3baa640 | ||
![]() |
2c4da9c634 | ||
![]() |
8de0313aca | ||
![]() |
143c5b17f9 | ||
![]() |
ec32fde60d | ||
![]() |
f2ac5d4995 | ||
![]() |
716962af03 | ||
![]() |
156efde6b9 | ||
![]() |
a9b7716dfe | ||
![]() |
76a67a6715 | ||
![]() |
c3defdcda4 | ||
![]() |
d8092aa916 | ||
![]() |
8b5a3d31aa | ||
![]() |
11761b66b2 | ||
![]() |
f83f107bd1 | ||
![]() |
f1b90857ff | ||
![]() |
5aea4cdd65 | ||
![]() |
1822a59c70 | ||
![]() |
37c6934802 | ||
![]() |
90c7f747a4 | ||
![]() |
b7efa3dcd6 | ||
![]() |
da80f537ac | ||
![]() |
0119912318 | ||
![]() |
5e0b1d0161 | ||
![]() |
35145d100b | ||
![]() |
584dd0ba89 | ||
![]() |
e5f5dc56c9 | ||
![]() |
880efb012a | ||
![]() |
b95308ac24 | ||
![]() |
3b192ad357 | ||
![]() |
b04938ac5e | ||
![]() |
56410ac1f2 | ||
![]() |
45a2eadc58 | ||
![]() |
0e03151505 | ||
![]() |
3e9706e6ce | ||
![]() |
a1646a7a88 | ||
![]() |
434511658b | ||
![]() |
04230fdd9c | ||
![]() |
2e16ad8a2a | ||
![]() |
7fb59b0fa9 | ||
![]() |
24e044bcda | ||
![]() |
1093656ec5 | ||
![]() |
674e997060 | ||
![]() |
37fd124f56 | ||
![]() |
7df2e67312 | ||
![]() |
8ea1f0796c | ||
![]() |
00fb5bc6cf | ||
![]() |
5e01f726d3 | ||
![]() |
e87290aeea | ||
![]() |
e7236beedd | ||
![]() |
5a6f54c96c | ||
![]() |
7769985439 | ||
![]() |
f4afdb21b2 | ||
![]() |
cc1b9e0a50 | ||
![]() |
2c58740cbb | ||
![]() |
808533b110 | ||
![]() |
e04a15bb99 | ||
![]() |
26d4074a61 | ||
![]() |
0137de5ca2 | ||
![]() |
b0a1b2edba | ||
![]() |
355c560ddc | ||
![]() |
72f286d9c6 | ||
![]() |
db6786414a | ||
![]() |
f8f046f193 | ||
![]() |
6c9ad4ded2 | ||
![]() |
32cae8ca11 | ||
![]() |
0cb65279d9 | ||
![]() |
412da06554 | ||
![]() |
247f8a2778 | ||
![]() |
017c9a6a70 | ||
![]() |
eccb2430b5 | ||
![]() |
5c64ebf1cc | ||
![]() |
1d7796805c | ||
![]() |
3479aaf6f6 | ||
![]() |
d2071cf05c | ||
![]() |
fb75508258 | ||
![]() |
0e69168382 | ||
![]() |
9970ab8643 | ||
![]() |
7ff82010c3 | ||
![]() |
1da8cd53d2 | ||
![]() |
9337a89eac | ||
![]() |
510564dd91 | ||
![]() |
a9ededc997 | ||
![]() |
722cffbd48 | ||
![]() |
a787befd72 | ||
![]() |
8ddd71a65a | ||
![]() |
280e1ec16d | ||
![]() |
52cf10cb2e | ||
![]() |
c12082affc | ||
![]() |
c0d7d74dac | ||
![]() |
df72b24ad2 | ||
![]() |
26bdd15400 | ||
![]() |
e13310b4e0 | ||
![]() |
e9d4218705 | ||
![]() |
56b80e4e60 | ||
![]() |
679f6471e6 | ||
![]() |
984179695c | ||
![]() |
5cca484a82 | ||
![]() |
d91e2296b0 | ||
![]() |
59768c16e2 | ||
![]() |
580efeae1a | ||
![]() |
0a3b508c8a | ||
![]() |
5e2f590b76 | ||
![]() |
ee94396afa | ||
![]() |
98764b6356 | ||
![]() |
f01cc4d0d0 | ||
![]() |
4c0f17429b | ||
![]() |
6a2ae1c572 | ||
![]() |
a2681ce6cc | ||
![]() |
4e43606932 | ||
![]() |
28c0c3207a | ||
![]() |
ecfa0e2b49 | ||
![]() |
ec3122f320 | ||
![]() |
7e4b4c973c | ||
![]() |
64753df2dd | ||
![]() |
75b36ab886 | ||
![]() |
208a78c748 | ||
![]() |
027249c575 | ||
![]() |
5ceb960205 | ||
![]() |
1cea156c5a | ||
![]() |
e1efa1ecbc | ||
![]() |
517de5e179 | ||
![]() |
825b62a2a9 | ||
![]() |
5ec817776c | ||
![]() |
3ad0519add | ||
![]() |
9868522341 | ||
![]() |
5b8d8c617f | ||
![]() |
a1f66061e3 | ||
![]() |
d8811933c9 | ||
![]() |
c9864dde8d | ||
![]() |
ca849d6649 | ||
![]() |
95a2354a86 | ||
![]() |
538e0ac48c | ||
![]() |
1c17411f04 | ||
![]() |
cbe0f89548 | ||
![]() |
615b071dcb | ||
![]() |
2d74f121aa | ||
![]() |
f65c180b8f | ||
![]() |
eb389c472d | ||
![]() |
befdf08035 | ||
![]() |
7a43ac865e | ||
![]() |
c43fcf9fbf | ||
![]() |
472a633010 | ||
![]() |
fc46acb2e4 | ||
![]() |
7fde66eb40 | ||
![]() |
b3cea3997d | ||
![]() |
2f20193086 | ||
![]() |
601e207f04 | ||
![]() |
f0d3d8b276 | ||
![]() |
e02d34f023 | ||
![]() |
0504be84b6 | ||
![]() |
216f020b70 | ||
![]() |
66505eedfa | ||
![]() |
b307c7d88b | ||
![]() |
47d4445fa8 | ||
![]() |
7098d2633a | ||
![]() |
61b5393b54 | ||
![]() |
6fe2cf13b6 | ||
![]() |
cc971eb2df | ||
![]() |
a260987f5c | ||
![]() |
5eb8f44dfc | ||
![]() |
e36131b38b | ||
![]() |
b67868afde | ||
![]() |
328fdee281 | ||
![]() |
8ab43e98fe | ||
![]() |
4f407ccc03 | ||
![]() |
5f3fddd3e4 | ||
![]() |
392f3b536d | ||
![]() |
155de2b396 | ||
![]() |
6c9930ac2a | ||
![]() |
9dddf4f4bc | ||
![]() |
314506c15c | ||
673222da6c | |||
![]() |
58b97e6361 | ||
![]() |
8bc2032b07 | ||
![]() |
40d4354dfc | ||
![]() |
27f2f5bb2b | ||
![]() |
490b934269 | ||
![]() |
e2e2b4d176 | ||
![]() |
7a1763bbee | ||
![]() |
ca8904b6bb | ||
![]() |
6384b28689 | ||
![]() |
e661466558 | ||
![]() |
8faf2beba4 | ||
![]() |
fb29319ec6 | ||
![]() |
7779d7d193 | ||
![]() |
2c5246b62f | ||
![]() |
e43a3f4518 | ||
![]() |
b5716f7a6d | ||
![]() |
fac5a3cc6f | ||
![]() |
671793078a | ||
![]() |
4f5ea1696f | ||
![]() |
b3e47e3bac | ||
![]() |
d67e990065 | ||
![]() |
7fb6f4b47f | ||
![]() |
df3b435a1f | ||
![]() |
5b65f1b544 | ||
![]() |
e159866d3b | ||
![]() |
aa0ce71612 | ||
![]() |
522e547f99 | ||
![]() |
9430341d4e | ||
![]() |
95838d045d | ||
![]() |
20832f11b6 | ||
![]() |
258203f400 | ||
![]() |
3b1768fa4e | ||
![]() |
23a3085bad | ||
![]() |
78c12e0ea6 | ||
![]() |
997cb85663 | ||
![]() |
968d5eabfa | ||
![]() |
590ce1f7ed | ||
![]() |
416a8838e4 | ||
![]() |
f6c349ac31 | ||
![]() |
517c6b40b5 | ||
![]() |
1fa78935a6 | ||
![]() |
4ddcd56740 | ||
![]() |
e5a2f33584 | ||
![]() |
c17cf3256a | ||
![]() |
5415bda02d | ||
![]() |
07cbaeb7aa | ||
![]() |
9d5aa83786 | ||
![]() |
463164cb88 | ||
![]() |
4809742088 | ||
![]() |
9cf0e285b4 | ||
![]() |
a6ca6f1cf9 | ||
![]() |
bd7479e1c0 | ||
![]() |
b35b51c203 | ||
![]() |
5652774888 | ||
![]() |
836e9a2fbc | ||
![]() |
64538c255d | ||
![]() |
62a9293dcf | ||
![]() |
1faae92f33 | ||
![]() |
dee808752f | ||
![]() |
5590af6995 | ||
![]() |
5afd161fba | ||
![]() |
336efa1e8b | ||
![]() |
568d5ca4ff | ||
![]() |
a9991f2a50 | ||
![]() |
1c8096444a | ||
![]() |
f424de46e6 | ||
![]() |
8fcca05565 | ||
![]() |
ed1f3cec59 | ||
![]() |
c29a409f28 | ||
![]() |
1a46936ad6 | ||
![]() |
e82abfb948 | ||
![]() |
c3409be780 | ||
![]() |
1557ab3474 | ||
![]() |
75fdda4fbf | ||
![]() |
4d75ef2849 | ||
![]() |
a1da23d1ba | ||
![]() |
c4bc751df7 | ||
![]() |
972d89ec6e | ||
![]() |
70f040e104 | ||
![]() |
d4de0b49c8 | ||
![]() |
2a42d58361 | ||
![]() |
14308956d7 | ||
![]() |
71339a66d7 | ||
![]() |
85f0664b56 | ||
![]() |
2f86ac0dfa | ||
![]() |
79d7b7a08d | ||
![]() |
b550d1990e | ||
![]() |
22062f0c77 | ||
![]() |
515f613a53 | ||
![]() |
615ed6b713 | ||
![]() |
f6eab62c3c | ||
![]() |
7d1e00ff0d | ||
![]() |
692439953c | ||
![]() |
6960a34d02 | ||
![]() |
b3662d4e6e | ||
![]() |
50c9b7c352 | ||
![]() |
f395d3b2bf | ||
![]() |
4fbf6cbc50 | ||
![]() |
9916958d6c | ||
![]() |
184a069c7f | ||
![]() |
0b65346633 | ||
![]() |
11f23dcc9e | ||
![]() |
f80d1defcb | ||
![]() |
e95d6343cb | ||
![]() |
a9052a068f | ||
![]() |
b1323c0d67 | ||
![]() |
87ceb8fb58 |
87
.github/workflows/runUiOctopusTests.yml
vendored
Normal file
87
.github/workflows/runUiOctopusTests.yml
vendored
Normal file
@@ -0,0 +1,87 @@
|
|||||||
|
name: Run Non Octopus UI Tests
|
||||||
|
on:
|
||||||
|
workflow_dispatch:
|
||||||
|
schedule:
|
||||||
|
- cron: '0 12 * * *'
|
||||||
|
jobs:
|
||||||
|
build-for-ui-test-mac-os:
|
||||||
|
if: github.repository == 'JetBrains/ideavim'
|
||||||
|
runs-on: macos-latest
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v4
|
||||||
|
- name: Apply Patch
|
||||||
|
run: |
|
||||||
|
git apply src/test/java/ui/octopus.patch
|
||||||
|
- name: Setup Java
|
||||||
|
uses: actions/setup-java@v4
|
||||||
|
with:
|
||||||
|
distribution: zulu
|
||||||
|
java-version: 11
|
||||||
|
- name: Setup FFmpeg
|
||||||
|
uses: FedericoCarboni/setup-ffmpeg@v3
|
||||||
|
with:
|
||||||
|
# Not strictly necessary, but it may prevent rate limit
|
||||||
|
# errors especially on GitHub-hosted macos machines.
|
||||||
|
github-token: ${{ secrets.GITHUB_TOKEN }}
|
||||||
|
- name: Setup Gradle
|
||||||
|
uses: gradle/gradle-build-action@v2.4.2
|
||||||
|
- name: Build Plugin
|
||||||
|
run: gradle :buildPlugin
|
||||||
|
- name: Run Idea
|
||||||
|
run: |
|
||||||
|
mkdir -p build/reports
|
||||||
|
gradle :runIdeForUiTests > build/reports/idea.log &
|
||||||
|
- name: Wait for Idea started
|
||||||
|
uses: jtalk/url-health-check-action@v3
|
||||||
|
with:
|
||||||
|
url: http://127.0.0.1:8082
|
||||||
|
max-attempts: 20
|
||||||
|
retry-delay: 10s
|
||||||
|
- name: Tests
|
||||||
|
run: gradle :testUi
|
||||||
|
- name: Move video
|
||||||
|
if: always()
|
||||||
|
run: mv video build/reports
|
||||||
|
- name: Move sandbox logs
|
||||||
|
if: always()
|
||||||
|
run: mv build/idea-sandbox/system/log sandbox-idea-log
|
||||||
|
- name: Save report
|
||||||
|
if: always()
|
||||||
|
uses: actions/upload-artifact@v4
|
||||||
|
with:
|
||||||
|
name: ui-test-fails-report-mac
|
||||||
|
path: |
|
||||||
|
build/reports
|
||||||
|
sandbox-idea-log
|
||||||
|
# build-for-ui-test-linux:
|
||||||
|
# runs-on: ubuntu-latest
|
||||||
|
# steps:
|
||||||
|
# - uses: actions/checkout@v2
|
||||||
|
# - name: Setup Java
|
||||||
|
# uses: actions/setup-java@v2.1.0
|
||||||
|
# with:
|
||||||
|
# distribution: zulu
|
||||||
|
# java-version: 11
|
||||||
|
# - name: Build Plugin
|
||||||
|
# run: gradle :buildPlugin
|
||||||
|
# - name: Run Idea
|
||||||
|
# run: |
|
||||||
|
# export DISPLAY=:99.0
|
||||||
|
# Xvfb -ac :99 -screen 0 1920x1080x16 &
|
||||||
|
# mkdir -p build/reports
|
||||||
|
# gradle :runIdeForUiTests #> build/reports/idea.log
|
||||||
|
# - name: Wait for Idea started
|
||||||
|
# uses: jtalk/url-health-check-action@1.5
|
||||||
|
# with:
|
||||||
|
# url: http://127.0.0.1:8082
|
||||||
|
# max-attempts: 15
|
||||||
|
# retry-delay: 30s
|
||||||
|
# - name: Tests
|
||||||
|
# run: gradle :testUi
|
||||||
|
# - name: Save fails report
|
||||||
|
# if: ${{ failure() }}
|
||||||
|
# uses: actions/upload-artifact@v2
|
||||||
|
# with:
|
||||||
|
# name: ui-test-fails-report-linux
|
||||||
|
# path: |
|
||||||
|
# ui-test-example/build/reports
|
24
.github/workflows/runUiTests.yml
vendored
24
.github/workflows/runUiTests.yml
vendored
@@ -8,18 +8,20 @@ jobs:
|
|||||||
if: github.repository == 'JetBrains/ideavim'
|
if: github.repository == 'JetBrains/ideavim'
|
||||||
runs-on: macos-latest
|
runs-on: macos-latest
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v2
|
- uses: actions/checkout@v4
|
||||||
- name: Setup Java
|
- name: Setup Java
|
||||||
uses: actions/setup-java@v2.1.0
|
uses: actions/setup-java@v4
|
||||||
with:
|
with:
|
||||||
distribution: zulu
|
distribution: zulu
|
||||||
java-version: 11
|
java-version: 11
|
||||||
- name: Setup FFmpeg
|
- name: Setup FFmpeg
|
||||||
uses: FedericoCarboni/setup-ffmpeg@v1
|
uses: FedericoCarboni/setup-ffmpeg@v3
|
||||||
with:
|
with:
|
||||||
# Not strictly necessary, but it may prevent rate limit
|
# Not strictly necessary, but it may prevent rate limit
|
||||||
# errors especially on GitHub-hosted macos machines.
|
# errors especially on GitHub-hosted macos machines.
|
||||||
token: ${{ secrets.GITHUB_TOKEN }}
|
github-token: ${{ secrets.GITHUB_TOKEN }}
|
||||||
|
- name: Setup Gradle
|
||||||
|
uses: gradle/gradle-build-action@v2.4.2
|
||||||
- name: Build Plugin
|
- name: Build Plugin
|
||||||
run: gradle :buildPlugin
|
run: gradle :buildPlugin
|
||||||
- name: Run Idea
|
- name: Run Idea
|
||||||
@@ -27,7 +29,7 @@ jobs:
|
|||||||
mkdir -p build/reports
|
mkdir -p build/reports
|
||||||
gradle :runIdeForUiTests > build/reports/idea.log &
|
gradle :runIdeForUiTests > build/reports/idea.log &
|
||||||
- name: Wait for Idea started
|
- name: Wait for Idea started
|
||||||
uses: jtalk/url-health-check-action@1.5
|
uses: jtalk/url-health-check-action@v3
|
||||||
with:
|
with:
|
||||||
url: http://127.0.0.1:8082
|
url: http://127.0.0.1:8082
|
||||||
max-attempts: 20
|
max-attempts: 20
|
||||||
@@ -35,15 +37,19 @@ jobs:
|
|||||||
- name: Tests
|
- name: Tests
|
||||||
run: gradle :testUi
|
run: gradle :testUi
|
||||||
- name: Move video
|
- name: Move video
|
||||||
if: ${{ failure() }}
|
if: always()
|
||||||
run: mv video build/reports
|
run: mv video build/reports
|
||||||
- name: Save fails report
|
- name: Move sandbox logs
|
||||||
if: ${{ failure() }}
|
if: always()
|
||||||
uses: actions/upload-artifact@v2
|
run: mv build/idea-sandbox/system/log sandbox-idea-log
|
||||||
|
- name: Save report
|
||||||
|
if: always()
|
||||||
|
uses: actions/upload-artifact@v4
|
||||||
with:
|
with:
|
||||||
name: ui-test-fails-report-mac
|
name: ui-test-fails-report-mac
|
||||||
path: |
|
path: |
|
||||||
build/reports
|
build/reports
|
||||||
|
sandbox-idea-log
|
||||||
# build-for-ui-test-linux:
|
# build-for-ui-test-linux:
|
||||||
# runs-on: ubuntu-latest
|
# runs-on: ubuntu-latest
|
||||||
# steps:
|
# steps:
|
||||||
|
1
.gitignore
vendored
1
.gitignore
vendored
@@ -23,6 +23,7 @@
|
|||||||
|
|
||||||
# Generated by gradle task "generateGrammarSource"
|
# Generated by gradle task "generateGrammarSource"
|
||||||
src/main/java/com/maddyhome/idea/vim/vimscript/parser/generated
|
src/main/java/com/maddyhome/idea/vim/vimscript/parser/generated
|
||||||
|
vim-engine/src/main/java/com/maddyhome/idea/vim/regexp/parser/generated
|
||||||
# Generated JSONs for lazy classloading
|
# Generated JSONs for lazy classloading
|
||||||
/vim-engine/src/main/resources/ksp-generated
|
/vim-engine/src/main/resources/ksp-generated
|
||||||
/src/main/resources/ksp-generated
|
/src/main/resources/ksp-generated
|
||||||
|
2
.idea/copyright/IdeaVim.xml
generated
2
.idea/copyright/IdeaVim.xml
generated
@@ -1,6 +1,6 @@
|
|||||||
<component name="CopyrightManager">
|
<component name="CopyrightManager">
|
||||||
<copyright>
|
<copyright>
|
||||||
<option name="notice" value="Copyright 2003-2023 The IdeaVim authors Use of this source code is governed by an MIT-style license that can be found in the LICENSE.txt file or at https://opensource.org/licenses/MIT." />
|
<option name="notice" value="Copyright 2003-&#36;today.year The IdeaVim authors Use of this source code is governed by an MIT-style license that can be found in the LICENSE.txt file or at https://opensource.org/licenses/MIT." />
|
||||||
<option name="myName" value="IdeaVim" />
|
<option name="myName" value="IdeaVim" />
|
||||||
</copyright>
|
</copyright>
|
||||||
</component>
|
</component>
|
16
.teamcity/_Self/Constants.kt
vendored
16
.teamcity/_Self/Constants.kt
vendored
@@ -5,13 +5,13 @@ object Constants {
|
|||||||
const val EAP_CHANNEL = "eap"
|
const val EAP_CHANNEL = "eap"
|
||||||
const val DEV_CHANNEL = "Dev"
|
const val DEV_CHANNEL = "Dev"
|
||||||
|
|
||||||
const val GITHUB_TESTS = "2023.1.2"
|
const val GITHUB_TESTS = "2023.3.2"
|
||||||
const val NVIM_TESTS = "2023.1.2"
|
const val NVIM_TESTS = "2023.3.2"
|
||||||
const val PROPERTY_TESTS = "2023.1.2"
|
const val PROPERTY_TESTS = "2023.3.2"
|
||||||
const val LONG_RUNNING_TESTS = "2023.1.2"
|
const val LONG_RUNNING_TESTS = "2023.3.2"
|
||||||
const val QODANA_TESTS = "2023.1.2"
|
const val QODANA_TESTS = "2023.3.2"
|
||||||
const val RELEASE = "2023.1.2"
|
const val RELEASE = "2023.3.2"
|
||||||
|
|
||||||
const val RELEASE_DEV = "2023.1.2"
|
const val RELEASE_DEV = "2023.3.2"
|
||||||
const val RELEASE_EAP = "2023.1.2"
|
const val RELEASE_EAP = "2023.3.2"
|
||||||
}
|
}
|
||||||
|
10
.teamcity/_Self/Project.kt
vendored
10
.teamcity/_Self/Project.kt
vendored
@@ -15,7 +15,7 @@ import jetbrains.buildServer.configs.kotlin.v2019_2.BuildType
|
|||||||
import jetbrains.buildServer.configs.kotlin.v2019_2.Project
|
import jetbrains.buildServer.configs.kotlin.v2019_2.Project
|
||||||
|
|
||||||
object Project : Project({
|
object Project : Project({
|
||||||
description = "Vim engine for IDEs based on the IntelliJ platform"
|
description = "Vim engine for JetBrains IDEs"
|
||||||
|
|
||||||
subProjects(Releases, OldTests, GitHub)
|
subProjects(Releases, OldTests, GitHub)
|
||||||
|
|
||||||
@@ -23,9 +23,8 @@ object Project : Project({
|
|||||||
vcsRoot(GitHubPullRequest)
|
vcsRoot(GitHubPullRequest)
|
||||||
|
|
||||||
// Active tests
|
// Active tests
|
||||||
buildType(TestingBuildType("2023.2", "<default>", version = "2023.2.3"))
|
|
||||||
buildType(TestingBuildType("2023.1", "<default>", version = "2023.1.5"))
|
|
||||||
buildType(TestingBuildType("Latest EAP", "<default>", version = "LATEST-EAP-SNAPSHOT"))
|
buildType(TestingBuildType("Latest EAP", "<default>", version = "LATEST-EAP-SNAPSHOT"))
|
||||||
|
buildType(TestingBuildType("2023.3", "<default>", version = "2023.3"))
|
||||||
buildType(TestingBuildType("Latest EAP With Xorg", "<default>", version = "LATEST-EAP-SNAPSHOT"))
|
buildType(TestingBuildType("Latest EAP With Xorg", "<default>", version = "LATEST-EAP-SNAPSHOT"))
|
||||||
|
|
||||||
buildType(PropertyBased)
|
buildType(PropertyBased)
|
||||||
@@ -40,6 +39,11 @@ object Project : Project({
|
|||||||
|
|
||||||
// Common build type for all configurations
|
// Common build type for all configurations
|
||||||
abstract class IdeaVimBuildType(init: BuildType.() -> Unit) : BuildType({
|
abstract class IdeaVimBuildType(init: BuildType.() -> Unit) : BuildType({
|
||||||
|
artifactRules = """
|
||||||
|
+:build/reports => build/reports
|
||||||
|
+:/mnt/agent/temp/buildTmp/ => /mnt/agent/temp/buildTmp/
|
||||||
|
""".trimIndent()
|
||||||
|
|
||||||
init()
|
init()
|
||||||
|
|
||||||
requirements {
|
requirements {
|
||||||
|
@@ -20,8 +20,8 @@ object PublishVimEngine : IdeaVimBuildType({
|
|||||||
params {
|
params {
|
||||||
param("env.ORG_GRADLE_PROJECT_engineVersion", "%build.number%")
|
param("env.ORG_GRADLE_PROJECT_engineVersion", "%build.number%")
|
||||||
param("env.ORG_GRADLE_PROJECT_uploadUrl", "https://packages.jetbrains.team/maven/p/ij/intellij-dependencies")
|
param("env.ORG_GRADLE_PROJECT_uploadUrl", "https://packages.jetbrains.team/maven/p/ij/intellij-dependencies")
|
||||||
password("env.ORG_GRADLE_PROJECT_spacePassword", "credentialsJSON:790b4e43-ee83-4184-b81b-678afab60409", display = ParameterDisplay.HIDDEN)
|
password("env.ORG_GRADLE_PROJECT_spacePassword", "credentialsJSON:5ea56f8c-efe7-4e1e-83de-0c02bcc39d0b", display = ParameterDisplay.HIDDEN)
|
||||||
param("env.ORG_GRADLE_PROJECT_spaceUsername", "Aleksei.Plate")
|
param("env.ORG_GRADLE_PROJECT_spaceUsername", "a121c67e-39ac-40e6-bf82-649855ec27b6")
|
||||||
}
|
}
|
||||||
|
|
||||||
vcs {
|
vcs {
|
||||||
|
16
.teamcity/_Self/buildTypes/ReleaseEap.kt
vendored
16
.teamcity/_Self/buildTypes/ReleaseEap.kt
vendored
@@ -5,6 +5,7 @@ import _Self.Constants.RELEASE_EAP
|
|||||||
import _Self.IdeaVimBuildType
|
import _Self.IdeaVimBuildType
|
||||||
import jetbrains.buildServer.configs.kotlin.v2019_2.CheckoutMode
|
import jetbrains.buildServer.configs.kotlin.v2019_2.CheckoutMode
|
||||||
import jetbrains.buildServer.configs.kotlin.v2019_2.DslContext
|
import jetbrains.buildServer.configs.kotlin.v2019_2.DslContext
|
||||||
|
import jetbrains.buildServer.configs.kotlin.v2019_2.ParameterDisplay
|
||||||
import jetbrains.buildServer.configs.kotlin.v2019_2.buildFeatures.sshAgent
|
import jetbrains.buildServer.configs.kotlin.v2019_2.buildFeatures.sshAgent
|
||||||
import jetbrains.buildServer.configs.kotlin.v2019_2.buildSteps.gradle
|
import jetbrains.buildServer.configs.kotlin.v2019_2.buildSteps.gradle
|
||||||
import jetbrains.buildServer.configs.kotlin.v2019_2.buildSteps.script
|
import jetbrains.buildServer.configs.kotlin.v2019_2.buildSteps.script
|
||||||
@@ -30,6 +31,11 @@ object ReleaseEap : IdeaVimBuildType({
|
|||||||
"credentialsJSON:a8ab8150-e6f8-4eaf-987c-bcd65eac50b5",
|
"credentialsJSON:a8ab8150-e6f8-4eaf-987c-bcd65eac50b5",
|
||||||
label = "Slack Token"
|
label = "Slack Token"
|
||||||
)
|
)
|
||||||
|
password(
|
||||||
|
"env.YOUTRACK_TOKEN",
|
||||||
|
"credentialsJSON:2479995b-7b60-4fbb-b095-f0bafae7f622",
|
||||||
|
display = ParameterDisplay.HIDDEN
|
||||||
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
vcs {
|
vcs {
|
||||||
@@ -61,13 +67,9 @@ object ReleaseEap : IdeaVimBuildType({
|
|||||||
tasks = "scripts:addReleaseTag"
|
tasks = "scripts:addReleaseTag"
|
||||||
}
|
}
|
||||||
gradle {
|
gradle {
|
||||||
|
name = "Publish plugin"
|
||||||
tasks = "publishPlugin"
|
tasks = "publishPlugin"
|
||||||
}
|
}
|
||||||
// Same as the script below. However, jgit can't find ssh keys on TeamCity
|
|
||||||
// gradle {
|
|
||||||
// name = "Push changes to the repo"
|
|
||||||
// tasks = "scripts:pushChanges"
|
|
||||||
// }
|
|
||||||
script {
|
script {
|
||||||
name = "Push changes to the repo"
|
name = "Push changes to the repo"
|
||||||
scriptContent = """
|
scriptContent = """
|
||||||
@@ -81,6 +83,10 @@ object ReleaseEap : IdeaVimBuildType({
|
|||||||
git push origin %build.number%
|
git push origin %build.number%
|
||||||
""".trimIndent()
|
""".trimIndent()
|
||||||
}
|
}
|
||||||
|
gradle {
|
||||||
|
name = "YouTrack post release actions"
|
||||||
|
tasks = "scripts:eapReleaseActions"
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
features {
|
features {
|
||||||
|
72
.teamcity/_Self/buildTypes/ReleasePlugin.kt
vendored
72
.teamcity/_Self/buildTypes/ReleasePlugin.kt
vendored
@@ -45,7 +45,11 @@ sealed class ReleasePlugin(private val releaseType: String) : IdeaVimBuildType({
|
|||||||
"credentialsJSON:a8ab8150-e6f8-4eaf-987c-bcd65eac50b5",
|
"credentialsJSON:a8ab8150-e6f8-4eaf-987c-bcd65eac50b5",
|
||||||
label = "Slack Token"
|
label = "Slack Token"
|
||||||
)
|
)
|
||||||
password("env.ORG_GRADLE_PROJECT_youtrackToken", "credentialsJSON:3cd3e867-282c-451f-b958-bc95d56a8450", display = ParameterDisplay.HIDDEN)
|
password(
|
||||||
|
"env.ORG_GRADLE_PROJECT_youtrackToken",
|
||||||
|
"credentialsJSON:7bc0eb3a-b86a-4ebd-b622-d4ef12d7e1d3",
|
||||||
|
display = ParameterDisplay.HIDDEN
|
||||||
|
)
|
||||||
param("env.ORG_GRADLE_PROJECT_releaseType", releaseType)
|
param("env.ORG_GRADLE_PROJECT_releaseType", releaseType)
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -65,9 +69,25 @@ sealed class ReleasePlugin(private val releaseType: String) : IdeaVimBuildType({
|
|||||||
name = "Pull git history"
|
name = "Pull git history"
|
||||||
scriptContent = "git fetch --unshallow"
|
scriptContent = "git fetch --unshallow"
|
||||||
}
|
}
|
||||||
gradle {
|
script {
|
||||||
name = "Select branch"
|
name = "Reset release branch"
|
||||||
tasks = "scripts:selectBranch"
|
scriptContent = """
|
||||||
|
if [ "major" = "$releaseType" ] || [ "minor" = "$releaseType" ]
|
||||||
|
then
|
||||||
|
echo Resetting the release branch because the release type is $releaseType
|
||||||
|
git checkout master
|
||||||
|
latest_eap=${'$'}(git describe --tags --match="[0-9].[0-9]*.[0-9]-eap.[0-9]*" --abbrev=0 HEAD)
|
||||||
|
echo Latest EAP: ${'$'}latest_eap
|
||||||
|
commit_of_latest_eap=${'$'}(git rev-list -n 1 ${'$'}latest_eap)
|
||||||
|
echo Commit of latest EAP: ${'$'}commit_of_latest_eap
|
||||||
|
git checkout release
|
||||||
|
git reset --hard ${'$'}commit_of_latest_eap
|
||||||
|
else
|
||||||
|
git checkout release
|
||||||
|
echo Do not reset the release branch because the release type is $releaseType
|
||||||
|
fi
|
||||||
|
echo Checked out release branch
|
||||||
|
""".trimIndent()
|
||||||
}
|
}
|
||||||
gradle {
|
gradle {
|
||||||
name = "Calculate new version"
|
name = "Calculate new version"
|
||||||
@@ -89,41 +109,47 @@ sealed class ReleasePlugin(private val releaseType: String) : IdeaVimBuildType({
|
|||||||
name = "Add release tag"
|
name = "Add release tag"
|
||||||
tasks = "scripts:addReleaseTag"
|
tasks = "scripts:addReleaseTag"
|
||||||
}
|
}
|
||||||
gradle {
|
script {
|
||||||
name = "Reset release branch"
|
name = "Run tests"
|
||||||
tasks = "scripts:resetReleaseBranch"
|
scriptContent = "./gradlew test"
|
||||||
}
|
}
|
||||||
gradle {
|
gradle {
|
||||||
name = "Publish release"
|
name = "Publish release"
|
||||||
tasks = "publishPlugin"
|
tasks = "publishPlugin"
|
||||||
}
|
}
|
||||||
// gradle {
|
script {
|
||||||
// name = "Push changes to the repo"
|
name = "Checkout master branch"
|
||||||
// tasks = "scripts:pushChangesWithReleaseBranch"
|
scriptContent = """
|
||||||
// }
|
echo Checkout master
|
||||||
|
git checkout master
|
||||||
|
""".trimIndent()
|
||||||
|
}
|
||||||
|
gradle {
|
||||||
|
name = "Update change log in master"
|
||||||
|
tasks = "scripts:changelogUpdateUnreleased"
|
||||||
|
}
|
||||||
|
gradle {
|
||||||
|
name = "Commit preparation changes in master"
|
||||||
|
tasks = "scripts:commitChanges"
|
||||||
|
}
|
||||||
script {
|
script {
|
||||||
name = "Push changes to the repo"
|
name = "Push changes to the repo"
|
||||||
scriptContent = """
|
scriptContent = """
|
||||||
branch=$(git branch --show-current)
|
branch=$(git branch --show-current)
|
||||||
echo current branch is ${'$'}branch
|
echo Current branch is ${'$'}branch
|
||||||
if [ "master" != "${'$'}branch" ];
|
if [ "master" != "${'$'}branch" ];
|
||||||
then
|
then
|
||||||
git checkout master
|
git checkout master
|
||||||
fi
|
fi
|
||||||
|
|
||||||
git push origin --tags
|
|
||||||
git push origin
|
git push origin
|
||||||
|
|
||||||
if [ "patch" != $releaseType ];
|
git checkout release
|
||||||
then
|
echo checkout release branch
|
||||||
git checkout release
|
git branch --set-upstream-to=origin/release release
|
||||||
echo checkout release branch
|
git push origin --force
|
||||||
git branch --set-upstream-to=origin/release release
|
# Push tag
|
||||||
git push --tags
|
git push origin %build.number%
|
||||||
git push origin --force
|
|
||||||
fi
|
|
||||||
|
|
||||||
git checkout ${'$'}branch
|
|
||||||
""".trimIndent()
|
""".trimIndent()
|
||||||
}
|
}
|
||||||
gradle {
|
gradle {
|
||||||
|
2
.teamcity/_Self/subprojects/OldTests.kt
vendored
2
.teamcity/_Self/subprojects/OldTests.kt
vendored
@@ -20,4 +20,6 @@ object OldTests : Project({
|
|||||||
buildType(TestingBuildType("IC-2021.2.2", "203-212", javaVersion = "1.8", javaPlugin = false))
|
buildType(TestingBuildType("IC-2021.2.2", "203-212", javaVersion = "1.8", javaPlugin = false))
|
||||||
buildType(TestingBuildType("IC-2021.3.2", "213-221", javaVersion = "1.8", javaPlugin = false))
|
buildType(TestingBuildType("IC-2021.3.2", "213-221", javaVersion = "1.8", javaPlugin = false))
|
||||||
buildType(TestingBuildType("IC-2022.2.3", branch = "222", javaPlugin = false))
|
buildType(TestingBuildType("IC-2022.2.3", branch = "222", javaPlugin = false))
|
||||||
|
buildType(TestingBuildType("IC-2023.1", "231-232", javaPlugin = false))
|
||||||
|
buildType(TestingBuildType("IC-2023.2", "231-232", javaPlugin = false))
|
||||||
})
|
})
|
||||||
|
29
.teamcity/patches/buildTypes/IdeaVimTests_Latest_EAP.kts
vendored
Normal file
29
.teamcity/patches/buildTypes/IdeaVimTests_Latest_EAP.kts
vendored
Normal file
@@ -0,0 +1,29 @@
|
|||||||
|
package patches.buildTypes
|
||||||
|
|
||||||
|
import jetbrains.buildServer.configs.kotlin.v2019_2.RelativeId
|
||||||
|
import jetbrains.buildServer.configs.kotlin.v2019_2.buildSteps.GradleBuildStep
|
||||||
|
import jetbrains.buildServer.configs.kotlin.v2019_2.buildSteps.gradle
|
||||||
|
import jetbrains.buildServer.configs.kotlin.v2019_2.ui.changeBuildType
|
||||||
|
import jetbrains.buildServer.configs.kotlin.v2019_2.ui.expectSteps
|
||||||
|
import jetbrains.buildServer.configs.kotlin.v2019_2.ui.update
|
||||||
|
|
||||||
|
/*
|
||||||
|
This patch script was generated by TeamCity on settings change in UI.
|
||||||
|
To apply the patch, change the buildType with id = 'IdeaVimTests_Latest_EAP'
|
||||||
|
accordingly, and delete the patch script.
|
||||||
|
*/
|
||||||
|
changeBuildType(RelativeId("IdeaVimTests_Latest_EAP")) {
|
||||||
|
expectSteps {
|
||||||
|
gradle {
|
||||||
|
tasks = "clean test"
|
||||||
|
buildFile = ""
|
||||||
|
enableStacktrace = true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
steps {
|
||||||
|
update<GradleBuildStep>(0) {
|
||||||
|
clearConditions()
|
||||||
|
jdkHome = "/usr/lib/jvm/java-17-amazon-corretto"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
20
.teamcity/patches/buildTypes/ReleaseMinor.kts
vendored
20
.teamcity/patches/buildTypes/ReleaseMinor.kts
vendored
@@ -1,20 +0,0 @@
|
|||||||
package patches.buildTypes
|
|
||||||
|
|
||||||
import jetbrains.buildServer.configs.kotlin.v2019_2.*
|
|
||||||
import jetbrains.buildServer.configs.kotlin.v2019_2.ui.*
|
|
||||||
|
|
||||||
/*
|
|
||||||
This patch script was generated by TeamCity on settings change in UI.
|
|
||||||
To apply the patch, change the buildType with id = 'ReleaseMinor'
|
|
||||||
accordingly, and delete the patch script.
|
|
||||||
*/
|
|
||||||
changeBuildType(RelativeId("ReleaseMinor")) {
|
|
||||||
params {
|
|
||||||
expect {
|
|
||||||
password("env.ORG_GRADLE_PROJECT_youtrackToken", "credentialsJSON:3cd3e867-282c-451f-b958-bc95d56a8450", display = ParameterDisplay.HIDDEN)
|
|
||||||
}
|
|
||||||
update {
|
|
||||||
password("env.ORG_GRADLE_PROJECT_youtrackToken", "credentialsJSON:7bc0eb3a-b86a-4ebd-b622-d4ef12d7e1d3", display = ParameterDisplay.HIDDEN)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
20
.teamcity/patches/buildTypes/ReleasePatch.kts
vendored
20
.teamcity/patches/buildTypes/ReleasePatch.kts
vendored
@@ -1,20 +0,0 @@
|
|||||||
package patches.buildTypes
|
|
||||||
|
|
||||||
import jetbrains.buildServer.configs.kotlin.v2019_2.*
|
|
||||||
import jetbrains.buildServer.configs.kotlin.v2019_2.ui.*
|
|
||||||
|
|
||||||
/*
|
|
||||||
This patch script was generated by TeamCity on settings change in UI.
|
|
||||||
To apply the patch, change the buildType with id = 'ReleasePatch'
|
|
||||||
accordingly, and delete the patch script.
|
|
||||||
*/
|
|
||||||
changeBuildType(RelativeId("ReleasePatch")) {
|
|
||||||
params {
|
|
||||||
expect {
|
|
||||||
password("env.ORG_GRADLE_PROJECT_youtrackToken", "credentialsJSON:3cd3e867-282c-451f-b958-bc95d56a8450", display = ParameterDisplay.HIDDEN)
|
|
||||||
}
|
|
||||||
update {
|
|
||||||
password("env.ORG_GRADLE_PROJECT_youtrackToken", "credentialsJSON:7bc0eb3a-b86a-4ebd-b622-d4ef12d7e1d3", display = ParameterDisplay.HIDDEN)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
2
.teamcity/settings.kts
vendored
2
.teamcity/settings.kts
vendored
@@ -30,5 +30,5 @@ node (Plugins -> teamcity-configs -> teamcity-configs:generate),
|
|||||||
the 'Debug' option is available in the context menu for the task.
|
the 'Debug' option is available in the context menu for the task.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
version = "2023.05"
|
version = "2023.11"
|
||||||
project(_Self.Project)
|
project(_Self.Project)
|
||||||
|
@@ -487,6 +487,14 @@ Contributors:
|
|||||||
[![icon][github]](https://github.com/pWydmuch)
|
[![icon][github]](https://github.com/pWydmuch)
|
||||||
|
|
||||||
pWydmuch
|
pWydmuch
|
||||||
|
* [![icon][mail]](mailto:leonid989@gmail.com)
|
||||||
|
[![icon][github]](https://github.com/Infonautica)
|
||||||
|
|
||||||
|
Leonid Danilov
|
||||||
|
* [![icon][mail]](mailto:emanuel-367@hotmail.com)
|
||||||
|
[![icon][github]](https://github.com/emanuelgestosa)
|
||||||
|
|
||||||
|
Emanuel Gestosa
|
||||||
|
|
||||||
Previous contributors:
|
Previous contributors:
|
||||||
|
|
||||||
|
30
CHANGES.md
30
CHANGES.md
@@ -25,12 +25,42 @@ usual beta standards.
|
|||||||
|
|
||||||
## To Be Released
|
## To Be Released
|
||||||
|
|
||||||
|
### Fixes:
|
||||||
|
* [VIM-3055](https://youtrack.jetbrains.com/issue/VIM-3055) Fix the issue with double deleting after dot
|
||||||
|
|
||||||
|
### Merged PRs:
|
||||||
|
* [725](https://github.com/JetBrains/ideavim/pull/725) by [Emanuel Gestosa](https://github.com/emanuelgestosa): Regex
|
||||||
|
|
||||||
|
## 2.8.0, 2024-01-30
|
||||||
|
|
||||||
### Fixes:
|
### Fixes:
|
||||||
* [VIM-3130](https://youtrack.jetbrains.com/issue/VIM-3130) Change the build version to 2023.1.2
|
* [VIM-3130](https://youtrack.jetbrains.com/issue/VIM-3130) Change the build version to 2023.1.2
|
||||||
* [VIM-3168](https://youtrack.jetbrains.com/issue/VIM-3168) Do not switch to block caret after enter if the IdeaVim is disabled
|
* [VIM-3168](https://youtrack.jetbrains.com/issue/VIM-3168) Do not switch to block caret after enter if the IdeaVim is disabled
|
||||||
* [VIM-3165](https://youtrack.jetbrains.com/issue/VIM-3165) Do not process enter key as IdeaVim shortcut if it's not an actual keypress
|
* [VIM-3165](https://youtrack.jetbrains.com/issue/VIM-3165) Do not process enter key as IdeaVim shortcut if it's not an actual keypress
|
||||||
* [VIM-3159](https://youtrack.jetbrains.com/issue/VIM-3159) Shift-enter now works in normal mode again
|
* [VIM-3159](https://youtrack.jetbrains.com/issue/VIM-3159) Shift-enter now works in normal mode again
|
||||||
* [VIM-3157](https://youtrack.jetbrains.com/issue/VIM-3157) Do not invoke enter in invokeLater for python console
|
* [VIM-3157](https://youtrack.jetbrains.com/issue/VIM-3157) Do not invoke enter in invokeLater for python console
|
||||||
|
* [VIM-3195](https://youtrack.jetbrains.com/issue/VIM-3195) Fix escape in injected editor
|
||||||
|
* [VIM-3190](https://youtrack.jetbrains.com/issue/VIM-3190) Do not use octopus handler if the enter key is used with modifiers like shift or control
|
||||||
|
* [VIM-3203](https://youtrack.jetbrains.com/issue/VIM-3203) Split action not works in normal mode
|
||||||
|
* [VIM-3184](https://youtrack.jetbrains.com/issue/VIM-3184) Revert "VIM-3184: Temporally disable new handlers for the thin client"
|
||||||
|
* [VIM-3186](https://youtrack.jetbrains.com/issue/VIM-3186) Do not multiply the enter action by the amount of carets
|
||||||
|
* [VIM-3177](https://youtrack.jetbrains.com/issue/VIM-3177) Formatting of commit message works again
|
||||||
|
* [VIM-1611](https://youtrack.jetbrains.com/issue/VIM-1611) actions related to resolving conflicts doesn't seem to work
|
||||||
|
* [VIM-3204](https://youtrack.jetbrains.com/issue/VIM-3204) Add checker that verifies the configuratin of the keymap
|
||||||
|
* [VIM-3084](https://youtrack.jetbrains.com/issue/VIM-3084) Double update for the status bar icon
|
||||||
|
* [VIM-3176](https://youtrack.jetbrains.com/issue/VIM-3176) Reselecting visual selection after pasting above it select wrong lines
|
||||||
|
* [VIM-3206](https://youtrack.jetbrains.com/issue/VIM-3206) Disable both copilot suggestion and insert mode on a single escape
|
||||||
|
* [VIM-3090](https://youtrack.jetbrains.com/issue/VIM-3090) Cmd line mode saves the visual mode
|
||||||
|
* [VIM-3085](https://youtrack.jetbrains.com/issue/VIM-3085) Open access to VimTypedActionHandler and VimShortcutKeyAction
|
||||||
|
* [VIM-3260](https://youtrack.jetbrains.com/issue/VIM-3260) Processing the offsets at the file end
|
||||||
|
* [VIM-3183](https://youtrack.jetbrains.com/issue/VIM-3183) Execute .ideavimrc on pooled thread
|
||||||
|
|
||||||
|
### Merged PRs:
|
||||||
|
* [763](https://github.com/JetBrains/ideavim/pull/763) by [Sam Ng](https://github.com/samabcde): Fix(VIM-3176) add test for restore selection after pasting in/below s…
|
||||||
|
* [772](https://github.com/JetBrains/ideavim/pull/772) by [chylex](https://github.com/chylex): Prevent code completion popup from appearing after running a macro
|
||||||
|
* [787](https://github.com/JetBrains/ideavim/pull/787) by [Leonid Danilov](https://github.com/Infonautica): Added "Which-Key" to Plugins
|
||||||
|
* [778](https://github.com/JetBrains/ideavim/pull/778) by [lippfi](https://github.com/lippfi): Showmode
|
||||||
|
* [788](https://github.com/JetBrains/ideavim/pull/788) by [Matt Ellis](https://github.com/citizenmatt): Refactor VimOptionGroupBase
|
||||||
|
|
||||||
## 2.7.0, 2023-11-07
|
## 2.7.0, 2023-11-07
|
||||||
|
|
||||||
|
@@ -324,7 +324,7 @@ IdeaVim tips and tricks
|
|||||||
- Use the power of IJ and Vim:
|
- Use the power of IJ and Vim:
|
||||||
- `set ideajoin` to enable join via the IDE. See the [examples](https://jb.gg/f9zji9).
|
- `set ideajoin` to enable join via the IDE. See the [examples](https://jb.gg/f9zji9).
|
||||||
- Make sure `ideaput` is enabled for `clipboard` to enable native IJ insertion in Vim.
|
- Make sure `ideaput` is enabled for `clipboard` to enable native IJ insertion in Vim.
|
||||||
- Sync IJ bookmarks and Vim marks: `set ideamarks`
|
- Sync IJ bookmarks and IdeaVim global marks: `set ideamarks` (works for marks with capital letters only)
|
||||||
- Check out more [ex commands](https://github.com/JetBrains/ideavim/wiki/%22set%22-commands).
|
- Check out more [ex commands](https://github.com/JetBrains/ideavim/wiki/%22set%22-commands).
|
||||||
|
|
||||||
- Use your vim settings with IdeaVim. Put `source ~/.vimrc` in `~/.ideavimrc`.
|
- Use your vim settings with IdeaVim. Put `source ~/.vimrc` in `~/.ideavimrc`.
|
||||||
|
@@ -84,3 +84,8 @@ IV) It is not allowed to remove this license from the distribution of the Vim
|
|||||||
license for previous Vim releases instead of the license that they came
|
license for previous Vim releases instead of the license that they came
|
||||||
with, at your option.
|
with, at your option.
|
||||||
```
|
```
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
File [sneakIcon.png](doc/images/sneakIcon.svg), which is originally an icon of the ideavim-sneak plugin,
|
||||||
|
is merged icons of IdeaVim plugin and a random sneaker by FreePic from flaticon.com.
|
||||||
|
@@ -8,9 +8,11 @@
|
|||||||
|
|
||||||
plugins {
|
plugins {
|
||||||
kotlin("jvm")
|
kotlin("jvm")
|
||||||
kotlin("plugin.serialization") version "1.8.21"
|
kotlin("plugin.serialization") version "1.9.22"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
val kotlinxSerializationVersion: String by project
|
||||||
|
|
||||||
group = "com.intellij"
|
group = "com.intellij"
|
||||||
version = "SNAPSHOT"
|
version = "SNAPSHOT"
|
||||||
|
|
||||||
@@ -19,6 +21,10 @@ repositories {
|
|||||||
}
|
}
|
||||||
|
|
||||||
dependencies {
|
dependencies {
|
||||||
compileOnly("com.google.devtools.ksp:symbol-processing-api:1.9.20-1.0.14")
|
compileOnly("com.google.devtools.ksp:symbol-processing-api:1.9.22-1.0.17")
|
||||||
implementation("org.jetbrains.kotlinx:kotlinx-serialization-json-jvm:1.6.0")
|
implementation("org.jetbrains.kotlinx:kotlinx-serialization-json-jvm:$kotlinxSerializationVersion") {
|
||||||
|
// kotlin stdlib is provided by IJ, so there is no need to include it into the distribution
|
||||||
|
exclude("org.jetbrains.kotlin", "kotlin-stdlib")
|
||||||
|
exclude("org.jetbrains.kotlin", "kotlin-stdlib-common")
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@@ -44,7 +44,7 @@ enum class Mode(val abbrev: Char) {
|
|||||||
OP_PENDING('O'),
|
OP_PENDING('O'),
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Indicates this key mapping applies to Insert mode
|
* Indicates this key mapping applies to Insert or Replace modes
|
||||||
*/
|
*/
|
||||||
INSERT('I'),
|
INSERT('I'),
|
||||||
|
|
||||||
|
@@ -44,19 +44,19 @@ buildscript {
|
|||||||
}
|
}
|
||||||
|
|
||||||
dependencies {
|
dependencies {
|
||||||
classpath("org.jetbrains.kotlin:kotlin-gradle-plugin:1.8.21")
|
classpath("org.jetbrains.kotlin:kotlin-gradle-plugin:1.9.22")
|
||||||
classpath("com.github.AlexPl292:mark-down-to-slack:1.1.2")
|
classpath("com.github.AlexPl292:mark-down-to-slack:1.1.2")
|
||||||
classpath("org.eclipse.jgit:org.eclipse.jgit:6.6.0.202305301015-r")
|
classpath("org.eclipse.jgit:org.eclipse.jgit:6.6.0.202305301015-r")
|
||||||
|
|
||||||
// This is needed for jgit to connect to ssh
|
// This is needed for jgit to connect to ssh
|
||||||
classpath("org.eclipse.jgit:org.eclipse.jgit.ssh.apache:6.7.0.202309050840-r")
|
classpath("org.eclipse.jgit:org.eclipse.jgit.ssh.apache:6.8.0.202311291450-r")
|
||||||
classpath("org.kohsuke:github-api:1.305")
|
classpath("org.kohsuke:github-api:1.305")
|
||||||
|
|
||||||
classpath("io.ktor:ktor-client-core:2.3.6")
|
classpath("io.ktor:ktor-client-core:2.3.7")
|
||||||
classpath("io.ktor:ktor-client-cio:2.3.5")
|
classpath("io.ktor:ktor-client-cio:2.3.7")
|
||||||
classpath("io.ktor:ktor-client-auth:2.3.6")
|
classpath("io.ktor:ktor-client-auth:2.3.7")
|
||||||
classpath("io.ktor:ktor-client-content-negotiation:2.3.6")
|
classpath("io.ktor:ktor-client-content-negotiation:2.3.7")
|
||||||
classpath("io.ktor:ktor-serialization-kotlinx-json:2.3.6")
|
classpath("io.ktor:ktor-serialization-kotlinx-json:2.3.7")
|
||||||
|
|
||||||
// This comes from the changelog plugin
|
// This comes from the changelog plugin
|
||||||
// classpath("org.jetbrains:markdown:0.3.1")
|
// classpath("org.jetbrains:markdown:0.3.1")
|
||||||
@@ -66,10 +66,11 @@ buildscript {
|
|||||||
plugins {
|
plugins {
|
||||||
antlr
|
antlr
|
||||||
java
|
java
|
||||||
kotlin("jvm") version "1.8.21"
|
kotlin("jvm") version "1.9.22"
|
||||||
application
|
application
|
||||||
|
id("java-test-fixtures")
|
||||||
|
|
||||||
id("org.jetbrains.intellij") version "1.16.0"
|
id("org.jetbrains.intellij") version "1.17.0"
|
||||||
id("org.jetbrains.changelog") version "2.2.0"
|
id("org.jetbrains.changelog") version "2.2.0"
|
||||||
|
|
||||||
// ktlint linter - read more: https://github.com/JLLeitschuh/ktlint-gradle
|
// ktlint linter - read more: https://github.com/JLLeitschuh/ktlint-gradle
|
||||||
@@ -78,7 +79,7 @@ plugins {
|
|||||||
id("org.jetbrains.kotlinx.kover") version "0.6.1"
|
id("org.jetbrains.kotlinx.kover") version "0.6.1"
|
||||||
id("com.dorongold.task-tree") version "2.1.1"
|
id("com.dorongold.task-tree") version "2.1.1"
|
||||||
|
|
||||||
id("com.google.devtools.ksp") version "1.8.21-1.0.11"
|
id("com.google.devtools.ksp") version "1.9.22-1.0.17"
|
||||||
}
|
}
|
||||||
|
|
||||||
ksp {
|
ksp {
|
||||||
@@ -91,6 +92,8 @@ ksp {
|
|||||||
afterEvaluate {
|
afterEvaluate {
|
||||||
// tasks.named("kspKotlin").configure { dependsOn("clean") }
|
// tasks.named("kspKotlin").configure { dependsOn("clean") }
|
||||||
tasks.named("kspKotlin").configure { dependsOn("generateGrammarSource") }
|
tasks.named("kspKotlin").configure { dependsOn("generateGrammarSource") }
|
||||||
|
tasks.named("kspTestFixturesKotlin").configure { enabled = false }
|
||||||
|
tasks.named("kspTestFixturesKotlin").configure { enabled = false }
|
||||||
tasks.named("kspTestKotlin").configure { enabled = false }
|
tasks.named("kspTestKotlin").configure { enabled = false }
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -98,6 +101,7 @@ afterEvaluate {
|
|||||||
val javaVersion: String by project
|
val javaVersion: String by project
|
||||||
val kotlinVersion: String by project
|
val kotlinVersion: String by project
|
||||||
val ideaVersion: String by project
|
val ideaVersion: String by project
|
||||||
|
val ideaType: String by project
|
||||||
val downloadIdeaSources: String by project
|
val downloadIdeaSources: String by project
|
||||||
val instrumentPluginCode: String by project
|
val instrumentPluginCode: String by project
|
||||||
val remoteRobotVersion: String by project
|
val remoteRobotVersion: String by project
|
||||||
@@ -115,35 +119,45 @@ repositories {
|
|||||||
}
|
}
|
||||||
|
|
||||||
dependencies {
|
dependencies {
|
||||||
|
api(project(":vim-engine"))
|
||||||
|
ksp(project(":annotation-processors"))
|
||||||
|
implementation(project(":annotation-processors"))
|
||||||
|
|
||||||
compileOnly("org.jetbrains.kotlin:kotlin-stdlib:$kotlinVersion")
|
compileOnly("org.jetbrains.kotlin:kotlin-stdlib:$kotlinVersion")
|
||||||
compileOnly("org.jetbrains:annotations:24.0.1")
|
compileOnly("org.jetbrains:annotations:24.1.0")
|
||||||
|
runtimeOnly("org.antlr:antlr4-runtime:$antlrVersion")
|
||||||
|
antlr("org.antlr:antlr4:$antlrVersion")
|
||||||
|
|
||||||
|
// --------- Test dependencies ----------
|
||||||
|
|
||||||
|
testImplementation(testFixtures(project(":")))
|
||||||
|
|
||||||
|
testApi("com.squareup.okhttp3:okhttp:4.12.0")
|
||||||
|
|
||||||
// https://mvnrepository.com/artifact/com.ensarsarajcic.neovim.java/neovim-api
|
// https://mvnrepository.com/artifact/com.ensarsarajcic.neovim.java/neovim-api
|
||||||
testImplementation("com.ensarsarajcic.neovim.java:neovim-api:0.2.3")
|
testImplementation("com.ensarsarajcic.neovim.java:neovim-api:0.2.3")
|
||||||
testImplementation("com.ensarsarajcic.neovim.java:core-rpc:0.2.3")
|
testImplementation("com.ensarsarajcic.neovim.java:core-rpc:0.2.3")
|
||||||
|
testFixturesImplementation("com.ensarsarajcic.neovim.java:neovim-api:0.2.3")
|
||||||
|
testFixturesImplementation("com.ensarsarajcic.neovim.java:core-rpc:0.2.3")
|
||||||
|
|
||||||
// https://mvnrepository.com/artifact/org.jetbrains.kotlin/kotlin-test
|
// https://mvnrepository.com/artifact/org.jetbrains.kotlin/kotlin-test
|
||||||
testImplementation("org.jetbrains.kotlin:kotlin-test:$kotlinVersion")
|
testImplementation("org.jetbrains.kotlin:kotlin-test:$kotlinVersion")
|
||||||
|
testFixturesImplementation("org.jetbrains.kotlin:kotlin-test:$kotlinVersion")
|
||||||
|
|
||||||
// https://mvnrepository.com/artifact/org.mockito.kotlin/mockito-kotlin
|
// https://mvnrepository.com/artifact/org.mockito.kotlin/mockito-kotlin
|
||||||
testImplementation("org.mockito.kotlin:mockito-kotlin:5.1.0")
|
testImplementation("org.mockito.kotlin:mockito-kotlin:5.2.1")
|
||||||
|
|
||||||
testImplementation("com.intellij.remoterobot:remote-robot:$remoteRobotVersion")
|
testImplementation("com.intellij.remoterobot:remote-robot:$remoteRobotVersion")
|
||||||
testImplementation("com.intellij.remoterobot:remote-fixtures:$remoteRobotVersion")
|
testImplementation("com.intellij.remoterobot:remote-fixtures:$remoteRobotVersion")
|
||||||
testImplementation("com.automation-remarks:video-recorder-junit:2.0")
|
testImplementation("com.intellij.remoterobot:ide-launcher:$remoteRobotVersion")
|
||||||
runtimeOnly("org.antlr:antlr4-runtime:$antlrVersion")
|
testImplementation("com.automation-remarks:video-recorder-junit5:2.0")
|
||||||
antlr("org.antlr:antlr4:$antlrVersion")
|
|
||||||
|
|
||||||
api(project(":vim-engine"))
|
testImplementation("org.junit.jupiter:junit-jupiter-api:5.10.1")
|
||||||
|
|
||||||
ksp(project(":annotation-processors"))
|
|
||||||
implementation(project(":annotation-processors"))
|
|
||||||
|
|
||||||
testApi("com.squareup.okhttp3:okhttp:4.12.0")
|
|
||||||
|
|
||||||
testImplementation("org.junit.jupiter:junit-jupiter-api:5.10.0")
|
|
||||||
testImplementation("org.junit.jupiter:junit-jupiter-engine:5.10.1")
|
testImplementation("org.junit.jupiter:junit-jupiter-engine:5.10.1")
|
||||||
testImplementation("org.junit.jupiter:junit-jupiter-params:5.10.1")
|
testImplementation("org.junit.jupiter:junit-jupiter-params:5.10.1")
|
||||||
|
testFixturesImplementation("org.junit.jupiter:junit-jupiter-api:5.10.1")
|
||||||
|
testFixturesImplementation("org.junit.jupiter:junit-jupiter-engine:5.10.1")
|
||||||
|
testFixturesImplementation("org.junit.jupiter:junit-jupiter-params:5.10.1")
|
||||||
}
|
}
|
||||||
|
|
||||||
configurations {
|
configurations {
|
||||||
@@ -184,6 +198,14 @@ tasks {
|
|||||||
include("**/*test.class")
|
include("**/*test.class")
|
||||||
include("**/*Tests.class")
|
include("**/*Tests.class")
|
||||||
exclude("**/ParserTest.class")
|
exclude("**/ParserTest.class")
|
||||||
|
|
||||||
|
// Set teamcity env variable locally to run additional tests for leaks.
|
||||||
|
// By default, this test runs on TC only, but this test doesn't take a lot of time,
|
||||||
|
// so we can turn it on for local development
|
||||||
|
if (environment["TEAMCITY_VERSION"] == null) {
|
||||||
|
println("Set env TEAMCITY_VERSION to X")
|
||||||
|
environment("TEAMCITY_VERSION" to "X")
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
val testWithNeovim by getting(Test::class) {
|
val testWithNeovim by getting(Test::class) {
|
||||||
@@ -224,7 +246,9 @@ tasks {
|
|||||||
compileKotlin {
|
compileKotlin {
|
||||||
kotlinOptions {
|
kotlinOptions {
|
||||||
jvmTarget = javaVersion
|
jvmTarget = javaVersion
|
||||||
apiVersion = "1.6"
|
// See https://plugins.jetbrains.com/docs/intellij/using-kotlin.html#kotlin-standard-library
|
||||||
|
// For the list of bundled versions
|
||||||
|
apiVersion = "1.9"
|
||||||
freeCompilerArgs = listOf("-Xjvm-default=all-compatibility")
|
freeCompilerArgs = listOf("-Xjvm-default=all-compatibility")
|
||||||
// allWarningsAsErrors = true
|
// allWarningsAsErrors = true
|
||||||
}
|
}
|
||||||
@@ -232,7 +256,7 @@ tasks {
|
|||||||
compileTestKotlin {
|
compileTestKotlin {
|
||||||
kotlinOptions {
|
kotlinOptions {
|
||||||
jvmTarget = javaVersion
|
jvmTarget = javaVersion
|
||||||
apiVersion = "1.6"
|
apiVersion = "1.9"
|
||||||
// allWarningsAsErrors = true
|
// allWarningsAsErrors = true
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -269,8 +293,7 @@ intellij {
|
|||||||
downloadSources.set(downloadIdeaSources.toBoolean())
|
downloadSources.set(downloadIdeaSources.toBoolean())
|
||||||
instrumentCode.set(instrumentPluginCode.toBoolean())
|
instrumentCode.set(instrumentPluginCode.toBoolean())
|
||||||
intellijRepository.set("https://www.jetbrains.com/intellij-repository")
|
intellijRepository.set("https://www.jetbrains.com/intellij-repository")
|
||||||
// Yaml is only used for testing. It's part of the IdeaIC distribution, but needs to be included as a reference
|
plugins.set(listOf("AceJump:3.8.11"))
|
||||||
plugins.set(listOf("java", "AceJump:3.8.11", "yaml"/*, "Pythonid:231.8109.2", "com.intellij.clion-swift:231.8109.4"*/))
|
|
||||||
}
|
}
|
||||||
|
|
||||||
tasks {
|
tasks {
|
||||||
@@ -294,6 +317,7 @@ tasks {
|
|||||||
systemProperty("ide.mac.message.dialogs.as.sheets", "false")
|
systemProperty("ide.mac.message.dialogs.as.sheets", "false")
|
||||||
systemProperty("jb.privacy.policy.text", "<!--999.999-->")
|
systemProperty("jb.privacy.policy.text", "<!--999.999-->")
|
||||||
systemProperty("jb.consents.confirmation.enabled", "false")
|
systemProperty("jb.consents.confirmation.enabled", "false")
|
||||||
|
systemProperty("ide.show.tips.on.startup.default.value", "false")
|
||||||
}
|
}
|
||||||
|
|
||||||
runPluginVerifier {
|
runPluginVerifier {
|
||||||
@@ -314,6 +338,9 @@ tasks {
|
|||||||
named("compileTestKotlin") {
|
named("compileTestKotlin") {
|
||||||
dependsOn("generateTestGrammarSource")
|
dependsOn("generateTestGrammarSource")
|
||||||
}
|
}
|
||||||
|
named("compileTestFixturesKotlin") {
|
||||||
|
dependsOn("generateTestFixturesGrammarSource")
|
||||||
|
}
|
||||||
|
|
||||||
// Add plugin open API sources to the plugin ZIP
|
// Add plugin open API sources to the plugin ZIP
|
||||||
val createOpenApiSourceJar by registering(Jar::class) {
|
val createOpenApiSourceJar by registering(Jar::class) {
|
||||||
@@ -344,7 +371,7 @@ tasks {
|
|||||||
val pluginVersion = version
|
val pluginVersion = version
|
||||||
// Don't forget to update plugin.xml
|
// Don't forget to update plugin.xml
|
||||||
patchPluginXml {
|
patchPluginXml {
|
||||||
sinceBuild.set("231.7515.13")
|
sinceBuild.set("233.11799.30")
|
||||||
|
|
||||||
// Get the latest available change notes from the changelog file
|
// Get the latest available change notes from the changelog file
|
||||||
changeNotes.set(
|
changeNotes.set(
|
||||||
@@ -420,6 +447,7 @@ kover {
|
|||||||
|
|
||||||
tasks.register("slackNotification") {
|
tasks.register("slackNotification") {
|
||||||
doLast {
|
doLast {
|
||||||
|
if (version.toString().last() != '0') return@doLast
|
||||||
if (slackUrl.isBlank()) {
|
if (slackUrl.isBlank()) {
|
||||||
println("Slack Url is not defined")
|
println("Slack Url is not defined")
|
||||||
return@doLast
|
return@doLast
|
||||||
@@ -524,10 +552,12 @@ tasks.register("releaseActions") {
|
|||||||
if (tickets.isNotEmpty()) {
|
if (tickets.isNotEmpty()) {
|
||||||
println("Updating statuses for tickets: $tickets")
|
println("Updating statuses for tickets: $tickets")
|
||||||
setYoutrackStatus(tickets, "Fixed")
|
setYoutrackStatus(tickets, "Fixed")
|
||||||
if (getVersionIdByName(version.toString()) != null) {
|
println("Checking if version $version exists...")
|
||||||
|
val versionId = getVersionIdByName(version.toString())
|
||||||
|
if (versionId == null) {
|
||||||
addReleaseToYoutrack(version.toString())
|
addReleaseToYoutrack(version.toString())
|
||||||
} else {
|
} else {
|
||||||
println("Version $version is already exists in YouTrack")
|
println("Version $version already exists in YouTrack. Version id: $versionId")
|
||||||
}
|
}
|
||||||
setYoutrackFixVersion(tickets, version.toString())
|
setYoutrackFixVersion(tickets, version.toString())
|
||||||
} else {
|
} else {
|
||||||
|
@@ -45,15 +45,20 @@ All commands with the mappings are supported. See the [full list of supported co
|
|||||||
<details>
|
<details>
|
||||||
<summary><h2>sneak</h2></summary>
|
<summary><h2>sneak</h2></summary>
|
||||||
|
|
||||||
|
<img src="images/sneakIcon.svg" width="80" height="80" alt="icon"/>
|
||||||
|
|
||||||
|
By [Mikhail Levchenko](https://github.com/Mishkun)
|
||||||
|
Original repository with the plugin: https://github.com/Mishkun/ideavim-sneak
|
||||||
Original plugin: [vim-sneak](https://github.com/justinmk/vim-sneak).
|
Original plugin: [vim-sneak](https://github.com/justinmk/vim-sneak).
|
||||||
|
|
||||||
### Setup:
|
### Setup:
|
||||||
- Install [IdeaVim-sneak](https://plugins.jetbrains.com/plugin/15348-ideavim-sneak) plugin.
|
- Add the following command to `~/.ideavimrc`: `Plug 'justinmk/vim-sneak'`
|
||||||
- Add the following command to `~/.ideavimrc`: `set sneak`
|
|
||||||
|
|
||||||
### Instructions
|
### Instructions
|
||||||
|
|
||||||
See the [docs](https://github.com/Mishkun/ideavim-sneak#usage)
|
* Type `s` and two chars to start sneaking in forward direction
|
||||||
|
* Type `S` and two chars to start sneaking in backward direction
|
||||||
|
* Type `;` or `,` to proceed with sneaking just as if you were using `f` or `t` commands
|
||||||
|
|
||||||
</details>
|
</details>
|
||||||
|
|
||||||
@@ -396,3 +401,19 @@ Original plugin: [quick-scope](https://github.com/unblevable/quick-scope).
|
|||||||
https://plugins.jetbrains.com/plugin/19417-ideavim-quickscope
|
https://plugins.jetbrains.com/plugin/19417-ideavim-quickscope
|
||||||
|
|
||||||
</details>
|
</details>
|
||||||
|
|
||||||
|
|
||||||
|
<details>
|
||||||
|
<summary><h2>Which-Key</h2></summary>
|
||||||
|
|
||||||
|
Original plugin: [vim-which-key](https://github.com/liuchengxu/vim-which-key).
|
||||||
|
|
||||||
|
### Setup:
|
||||||
|
- Install [Which-Key](https://plugins.jetbrains.com/plugin/15976-which-key) plugin.
|
||||||
|
- Add the following command to `~/.ideavimrc`: `set which-key`
|
||||||
|
|
||||||
|
### Instructions
|
||||||
|
|
||||||
|
https://github.com/TheBlob42/idea-which-key?tab=readme-ov-file#installation
|
||||||
|
|
||||||
|
</details>
|
||||||
|
@@ -3,6 +3,11 @@ Put `set ideajoin` to your `~/.ideavimrc` to enable this functionality.
|
|||||||
|
|
||||||
Now, you can press `J` (`shift+j`) on a line or a selected block of text to join the lines together.
|
Now, you can press `J` (`shift+j`) on a line or a selected block of text to join the lines together.
|
||||||
|
|
||||||
|
:warning: This feature is language-specific. This means that the IDE should implement this feature for a particular
|
||||||
|
language in order for the IDE to work as described below. If any of the examples provided below don't match your case,
|
||||||
|
please file an issue in the project related to your IDE: https://youtrack.jetbrains.com/.
|
||||||
|
Here is a list of known requests: https://youtrack.jetbrains.com/issues?q=links:VIM-3214.
|
||||||
|
|
||||||
* Automatic join concatenated lines:
|
* Automatic join concatenated lines:
|
||||||
|
|
||||||
```
|
```
|
||||||
|
28
doc/images/sneakIcon.svg
Normal file
28
doc/images/sneakIcon.svg
Normal file
@@ -0,0 +1,28 @@
|
|||||||
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
|
<svg viewBox="386.498 234 32 32" xmlns="http://www.w3.org/2000/svg">
|
||||||
|
<defs>
|
||||||
|
<linearGradient id="ideavim_plugin-a" x1="-6.748%" x2="47.286%" y1="33.61%" y2="85.907%">
|
||||||
|
<stop offset="0" stop-color="#3BEA62"/>
|
||||||
|
<stop offset="1" stop-color="#087CFA"/>
|
||||||
|
</linearGradient>
|
||||||
|
</defs>
|
||||||
|
<g transform="matrix(1.238978, 0.90017, -0.90017, 1.238978, 131.776901, -422.953003)" style="">
|
||||||
|
<path d="M 399.962 247.648 C 399.207 246.894 399.147 246.318 399.692 245.453 C 400.237 244.588 401.955 245.886 401.955 245.886 L 401.955 250.737" style="fill: rgb(248, 245, 231);"/>
|
||||||
|
<path d="M 413.846 253.602 C 413.846 255.077 411.827 256.134 409.392 256.134 L 396.381 256.134 C 395.211 256.134 394.232 256.003 393.433 255.817 C 391.496 255.367 390.613 254.596 390.613 254.596 L 391.478 252.513 L 393.607 252.617 Z M 413.846 253.602" fill="#cce6f6" style=""/>
|
||||||
|
<path d="M 413.846 253.602 C 413.846 253.602 411.475 254.468 408.27 254.468 C 405.94 254.468 398.116 253.433 394.023 252.869 C 392.488 252.658 391.478 252.512 391.478 252.512 C 390.864 251.662 392.078 248.741 392.758 247.263 C 393.118 246.484 393.85 245.929 394.703 245.83 C 394.782 245.82 394.861 245.815 394.939 245.815 C 395.369 245.815 395.645 246.532 396.059 247.225 C 396.446 247.877 396.955 248.507 397.823 248.507 C 399.617 248.507 401.955 245.886 401.955 245.886 C 406.544 249.03 410.097 250.43 410.097 250.43 C 410.097 250.43 413.061 250.446 413.782 251.167 C 414.503 251.888 414.422 253.218 413.846 253.602 Z M 413.846 253.602" style="fill: rgb(8, 124, 250);"/>
|
||||||
|
<path d="M 394.023 252.869 L 393.433 255.817 C 391.496 255.367 390.613 254.596 390.613 254.596 L 391.478 252.513 L 393.607 252.617 Z M 394.023 252.869" fill="#9dcae0" style=""/>
|
||||||
|
<path d="M 396.059 247.225 C 395.073 245.986 393.193 250.255 394.023 252.869 C 392.488 252.658 391.478 252.513 391.478 252.513 C 390.864 251.662 392.078 248.741 392.758 247.263 C 393.118 246.484 393.85 245.929 394.703 245.83 C 394.782 245.82 394.861 245.815 394.939 245.815 C 395.369 245.815 395.645 246.532 396.059 247.225 Z M 396.059 247.225" style="fill: rgb(14, 112, 142);"/>
|
||||||
|
<path d="M 403.527 246.924 L 399.174 251.768 C 397.7 250.892 395.578 250.174 394.016 249.717 C 392.843 249.372 391.985 249.174 391.959 249.168 C 392.108 248.769 392.268 248.379 392.421 248.022 L 394.341 248.65 L 394.884 248.827 C 395.509 249.031 396.192 248.95 396.753 248.608 L 397.153 248.363 C 397.35 248.454 397.572 248.507 397.823 248.507 C 399.617 248.507 401.955 245.886 401.955 245.886 C 402.494 246.256 403.02 246.601 403.527 246.924 Z M 403.527 246.924" style="fill: rgb(59, 234, 98);"/>
|
||||||
|
<path d="M 413.847 253.602 C 413.847 253.602 411.475 254.468 408.27 254.468 C 407.586 254.468 406.426 254.378 405.025 254.238 L 405.025 254.237 C 405.025 253.495 406.924 251.743 408.366 251.616 C 408.623 252.865 410.097 252.512 411.219 252.128 C 412.341 251.743 413.783 251.167 413.783 251.167 C 414.503 251.888 414.422 253.218 413.847 253.602 Z M 413.847 253.602" style="fill: rgb(8, 124, 250);"/>
|
||||||
|
<path d="M 394.341 248.65 C 394.214 248.978 394.103 249.339 394.016 249.717 C 392.843 249.372 391.985 249.174 391.959 249.168 C 392.108 248.769 392.268 248.379 392.421 248.022 Z M 394.341 248.65" style="fill: rgb(37, 187, 163);"/>
|
||||||
|
<path d="M 408.366 251.616 C 406.924 251.743 405.025 253.495 405.025 254.237 L 405.025 254.238 C 403.784 254.113 402.355 253.948 400.899 253.77 C 400.899 253.051 400.191 252.372 399.174 251.768 L 399.174 251.768 L 403.528 246.924 C 407.331 249.34 410.097 250.43 410.097 250.43 C 410.77 251.102 409.809 251.487 408.366 251.616 Z M 408.366 251.616" style="fill: rgb(248, 245, 231);"/>
|
||||||
|
<polygon fill="url(#ideavim_plugin-a)" fill-rule="evenodd" points="406.356 248.463 403.261 252.616 402.183 248.212 400.984 249.899 401.999 254.236 403.927 254.176 407.639 249.062" style="" transform="matrix(0.994522, 0.104529, -0.104529, 0.994522, 28.475005, -40.88594)"/>
|
||||||
|
<g fill="#fb6572" transform="matrix(0.046265, 0, 0, 0.046265, 390.612823, 245.155533)" style="">
|
||||||
|
<path d="m288.839844 72.65625c-2.007813 0-4.011719-.761719-5.542969-2.292969-3.058594-3.0625-3.058594-8.023437 0-11.082031l20.089844-20.089844c3.0625-3.058594 8.023437-3.058594 11.082031 0 3.058594 3.0625 3.0625 8.023438 0 11.082032l-20.089844 20.089843c-1.527344 1.53125-3.535156 2.292969-5.539062 2.292969zm0 0" style="fill: rgb(56, 228, 105);"/>
|
||||||
|
<path d="m314.589844 87.082031c-2.007813 0-4.011719-.765625-5.542969-2.296875-3.0625-3.058594-3.0625-8.019531 0-11.082031l20.089844-20.085937c3.0625-3.058594 8.023437-3.058594 11.082031 0 3.058594 3.0625 3.058594 8.023437 0 11.082031l-20.089844 20.085937c-1.527344 1.53125-3.535156 2.296875-5.539062 2.296875zm0 0" style="fill: rgb(59, 233, 100);"/>
|
||||||
|
<path d="m340.339844 101.507812c-2.007813 0-4.011719-.765624-5.542969-2.296874-3.058594-3.058594-3.058594-8.023438 0-11.082032l20.089844-20.085937c3.0625-3.0625 8.023437-3.0625 11.082031 0 3.058594 3.058593 3.0625 8.023437 0 11.082031l-20.089844 20.085938c-1.527344 1.53125-3.535156 2.296874-5.539062 2.296874zm0 0" style="fill: rgb(59, 233, 100);"/>
|
||||||
|
<path d="m366.089844 115.929688c-2.003906 0-4.011719-.761719-5.539063-2.292969-3.0625-3.0625-3.0625-8.023438 0-11.082031l20.085938-20.089844c3.0625-3.058594 8.023437-3.058594 11.082031 0 3.0625 3.0625 3.0625 8.023437 0 11.082031l-20.085938 20.089844c-1.53125 1.53125-3.535156 2.292969-5.542968 2.292969zm0 0" style="fill: rgb(59, 233, 100);"/>
|
||||||
|
</g>
|
||||||
|
<path d="M 401.925 247.748 C 401.761 247.748 401.611 247.629 401.573 247.469 C 401.536 247.312 401.611 247.144 401.753 247.067 C 402 246.933 402.318 247.147 402.286 247.426 C 402.265 247.606 402.107 247.748 401.925 247.748 Z M 401.925 247.748" fill="#1e2628" style=""/>
|
||||||
|
</g>
|
||||||
|
</svg>
|
After Width: | Height: | Size: 5.6 KiB |
@@ -8,20 +8,27 @@
|
|||||||
|
|
||||||
# suppress inspection "UnusedProperty" for whole file
|
# suppress inspection "UnusedProperty" for whole file
|
||||||
|
|
||||||
ideaVersion=2023.1.2
|
ideaVersion=2023.3.2
|
||||||
|
# Values for type: https://plugins.jetbrains.com/docs/intellij/tools-gradle-intellij-plugin.html#intellij-extension-type
|
||||||
|
ideaType=IC
|
||||||
downloadIdeaSources=true
|
downloadIdeaSources=true
|
||||||
instrumentPluginCode=true
|
instrumentPluginCode=true
|
||||||
version=SNAPSHOT
|
version=SNAPSHOT
|
||||||
javaVersion=17
|
javaVersion=17
|
||||||
remoteRobotVersion=0.11.17
|
remoteRobotVersion=0.11.21
|
||||||
antlrVersion=4.10.1
|
antlrVersion=4.10.1
|
||||||
|
|
||||||
|
|
||||||
# Please don't forget to update kotlin version in buildscript section
|
# Please don't forget to update kotlin version in buildscript section
|
||||||
kotlinVersion=1.8.21
|
# Also update kotlinxSerializationVersion version
|
||||||
|
kotlinVersion=1.9.22
|
||||||
publishToken=token
|
publishToken=token
|
||||||
publishChannels=eap
|
publishChannels=eap
|
||||||
|
|
||||||
|
# Kotlinx serialization also uses some version of kotlin stdlib under the hood. However,
|
||||||
|
# we exclude this version from the dependency and use our own version of kotlin that is specified above
|
||||||
|
kotlinxSerializationVersion=1.5.1
|
||||||
|
|
||||||
slackUrl=
|
slackUrl=
|
||||||
youtrackToken=
|
youtrackToken=
|
||||||
|
|
||||||
|
@@ -20,17 +20,17 @@ repositories {
|
|||||||
}
|
}
|
||||||
|
|
||||||
dependencies {
|
dependencies {
|
||||||
compileOnly("org.jetbrains.kotlin:kotlin-stdlib:1.9.20")
|
compileOnly("org.jetbrains.kotlin:kotlin-stdlib:1.9.22")
|
||||||
|
|
||||||
implementation("io.ktor:ktor-client-core:2.3.6")
|
implementation("io.ktor:ktor-client-core:2.3.7")
|
||||||
implementation("io.ktor:ktor-client-cio:2.3.5")
|
implementation("io.ktor:ktor-client-cio:2.3.7")
|
||||||
implementation("io.ktor:ktor-client-content-negotiation:2.3.6")
|
implementation("io.ktor:ktor-client-content-negotiation:2.3.7")
|
||||||
implementation("io.ktor:ktor-serialization-kotlinx-json:2.3.6")
|
implementation("io.ktor:ktor-serialization-kotlinx-json:2.3.7")
|
||||||
implementation("io.ktor:ktor-client-auth:2.3.6")
|
implementation("io.ktor:ktor-client-auth:2.3.7")
|
||||||
implementation("org.eclipse.jgit:org.eclipse.jgit:6.6.0.202305301015-r")
|
implementation("org.eclipse.jgit:org.eclipse.jgit:6.6.0.202305301015-r")
|
||||||
|
|
||||||
// This is needed for jgit to connect to ssh
|
// This is needed for jgit to connect to ssh
|
||||||
implementation("org.eclipse.jgit:org.eclipse.jgit.ssh.apache:6.7.0.202309050840-r")
|
implementation("org.eclipse.jgit:org.eclipse.jgit.ssh.apache:6.8.0.202311291450-r")
|
||||||
implementation("com.vdurmont:semver4j:3.1.0")
|
implementation("com.vdurmont:semver4j:3.1.0")
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -93,28 +93,16 @@ tasks.register("addReleaseTag", JavaExec::class) {
|
|||||||
args = listOf(project.version.toString(), rootProject.rootDir.toString(), releaseType ?: "")
|
args = listOf(project.version.toString(), rootProject.rootDir.toString(), releaseType ?: "")
|
||||||
}
|
}
|
||||||
|
|
||||||
tasks.register("resetReleaseBranch", JavaExec::class) {
|
tasks.register("selectBranch", JavaExec::class) {
|
||||||
group = "release"
|
group = "release"
|
||||||
mainClass.set("scripts.release.ResetReleaseBranchKt")
|
mainClass.set("scripts.release.SelectBranchKt")
|
||||||
classpath = sourceSets["main"].runtimeClasspath
|
classpath = sourceSets["main"].runtimeClasspath
|
||||||
args = listOf(project.version.toString(), rootProject.rootDir.toString(), releaseType ?: "")
|
args = listOf(project.version.toString(), rootProject.rootDir.toString(), releaseType ?: "")
|
||||||
}
|
}
|
||||||
|
|
||||||
tasks.register("pushChanges", JavaExec::class) {
|
tasks.register("eapReleaseActions", JavaExec::class) {
|
||||||
mainClass.set("scripts.PushChangesKt")
|
|
||||||
classpath = sourceSets["main"].runtimeClasspath
|
|
||||||
args = listOf(rootProject.rootDir.toString())
|
|
||||||
}
|
|
||||||
|
|
||||||
tasks.register("pushChangesWithReleaseBranch", JavaExec::class) {
|
|
||||||
mainClass.set("scripts.PushChangesWithReleaseBranchKt")
|
|
||||||
classpath = sourceSets["main"].runtimeClasspath
|
|
||||||
args = listOf(rootProject.rootDir.toString(), releaseType ?: "")
|
|
||||||
}
|
|
||||||
|
|
||||||
tasks.register("selectBranch", JavaExec::class) {
|
|
||||||
group = "release"
|
group = "release"
|
||||||
mainClass.set("scripts.release.SelectBranchKt")
|
mainClass.set("scripts.releaseEap.EapReleaseActionsKt")
|
||||||
classpath = sourceSets["main"].runtimeClasspath
|
classpath = sourceSets["main"].runtimeClasspath
|
||||||
args = listOf(project.version.toString(), rootProject.rootDir.toString(), releaseType ?: "")
|
args = listOf(project.version.toString(), rootProject.rootDir.toString(), releaseType ?: "")
|
||||||
}
|
}
|
||||||
|
@@ -32,6 +32,7 @@ val knownPlugins = listOf(
|
|||||||
"com.github.dankinsoid.multicursor",
|
"com.github.dankinsoid.multicursor",
|
||||||
"com.joshestein.ideavim-quickscope",
|
"com.joshestein.ideavim-quickscope",
|
||||||
"ca.alexgirard.HarpoonIJ",
|
"ca.alexgirard.HarpoonIJ",
|
||||||
|
"com.protoseo.input-source-auto-converter",
|
||||||
|
|
||||||
// "cc.implicated.intellij.plugins.bunny", // I don't want to include this plugin in the list of IdeaVim plugins as I don't understand what this is for
|
// "cc.implicated.intellij.plugins.bunny", // I don't want to include this plugin in the list of IdeaVim plugins as I don't understand what this is for
|
||||||
)
|
)
|
||||||
|
@@ -32,7 +32,7 @@ fun httpClient(): HttpClient {
|
|||||||
install(Auth) {
|
install(Auth) {
|
||||||
bearer {
|
bearer {
|
||||||
loadTokens {
|
loadTokens {
|
||||||
val accessToken = System.getenv("YOUTRACK_TOKEN")!!
|
val accessToken = System.getenv("YOUTRACK_TOKEN") ?: error("Missing YOUTRACK_TOKEN")
|
||||||
BearerTokens(accessToken, "")
|
BearerTokens(accessToken, "")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -1,37 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright 2003-2023 The IdeaVim authors
|
|
||||||
*
|
|
||||||
* Use of this source code is governed by an MIT-style
|
|
||||||
* license that can be found in the LICENSE.txt file or at
|
|
||||||
* https://opensource.org/licenses/MIT.
|
|
||||||
*/
|
|
||||||
|
|
||||||
package scripts
|
|
||||||
|
|
||||||
import scripts.release.checkoutBranch
|
|
||||||
import scripts.release.withGit
|
|
||||||
import scripts.release.withRepo
|
|
||||||
|
|
||||||
fun main(args: Array<String>) {
|
|
||||||
val rootDir = args[0]
|
|
||||||
println("root dir: $rootDir")
|
|
||||||
|
|
||||||
val currentBranch = withRepo(rootDir) { it.branch }
|
|
||||||
println("Current branch is $currentBranch")
|
|
||||||
|
|
||||||
|
|
||||||
withGit(rootDir) { git ->
|
|
||||||
if (currentBranch != "master") {
|
|
||||||
git.checkoutBranch("master")
|
|
||||||
println("Check out master branch")
|
|
||||||
}
|
|
||||||
|
|
||||||
git.push()
|
|
||||||
.setPushTags()
|
|
||||||
.call()
|
|
||||||
println("Master pushed with tags")
|
|
||||||
|
|
||||||
git.checkoutBranch(currentBranch)
|
|
||||||
println("Checked out $currentBranch branch")
|
|
||||||
}
|
|
||||||
}
|
|
@@ -1,54 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright 2003-2023 The IdeaVim authors
|
|
||||||
*
|
|
||||||
* Use of this source code is governed by an MIT-style
|
|
||||||
* license that can be found in the LICENSE.txt file or at
|
|
||||||
* https://opensource.org/licenses/MIT.
|
|
||||||
*/
|
|
||||||
|
|
||||||
package scripts
|
|
||||||
|
|
||||||
import scripts.release.checkoutBranch
|
|
||||||
import scripts.release.withGit
|
|
||||||
import scripts.release.withRepo
|
|
||||||
|
|
||||||
fun main(args: Array<String>) {
|
|
||||||
val rootDir = args[0]
|
|
||||||
val releaseType = args[1]
|
|
||||||
println("root dir: $rootDir")
|
|
||||||
println("releaseType: $releaseType")
|
|
||||||
|
|
||||||
val currentBranch = withRepo(rootDir) { it.branch }
|
|
||||||
println("Current branch is $currentBranch")
|
|
||||||
|
|
||||||
|
|
||||||
withGit(rootDir) { git ->
|
|
||||||
if (currentBranch != "master") {
|
|
||||||
git.checkoutBranch("master")
|
|
||||||
println("Check out master branch")
|
|
||||||
}
|
|
||||||
|
|
||||||
git.push()
|
|
||||||
.setPushTags()
|
|
||||||
.call()
|
|
||||||
println("Master pushed with tags")
|
|
||||||
|
|
||||||
if (releaseType != "patch") {
|
|
||||||
git.checkoutBranch("release")
|
|
||||||
println("Checked out release")
|
|
||||||
|
|
||||||
git
|
|
||||||
.push()
|
|
||||||
.setForce(true)
|
|
||||||
.setPushTags()
|
|
||||||
.call()
|
|
||||||
println("Pushed release branch with tags")
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
println("Do not push release branch because type of release is $releaseType")
|
|
||||||
}
|
|
||||||
|
|
||||||
git.checkoutBranch(currentBranch)
|
|
||||||
println("Checked out $currentBranch branch")
|
|
||||||
}
|
|
||||||
}
|
|
@@ -12,7 +12,7 @@ fun main(args: Array<String>) {
|
|||||||
println("HI!")
|
println("HI!")
|
||||||
val projectDir = args[0]
|
val projectDir = args[0]
|
||||||
println("Working directory: $projectDir")
|
println("Working directory: $projectDir")
|
||||||
val (lastVersion, objectId) = getVersion(projectDir, onlyStable = true)
|
val (lastVersion, objectId) = getVersion(projectDir, ReleaseType.STABLE_NO_PATCH)
|
||||||
println("Last version: $lastVersion, hash: ${objectId.name}")
|
println("Last version: $lastVersion, hash: ${objectId.name}")
|
||||||
|
|
||||||
val branch = withRepo(projectDir) { it.branch }
|
val branch = withRepo(projectDir) { it.branch }
|
||||||
|
@@ -12,7 +12,7 @@ fun main(args: Array<String>) {
|
|||||||
println("HI!")
|
println("HI!")
|
||||||
val projectDir = args[0]
|
val projectDir = args[0]
|
||||||
println("Working directory: $projectDir")
|
println("Working directory: $projectDir")
|
||||||
val (lastVersion, _) = getVersion(projectDir, onlyStable = false)
|
val (lastVersion, _) = getVersion(projectDir, ReleaseType.ANY)
|
||||||
|
|
||||||
val nextVersion = if (lastVersion.suffixTokens.isEmpty()) {
|
val nextVersion = if (lastVersion.suffixTokens.isEmpty()) {
|
||||||
lastVersion.nextMinor().withSuffix("eap.1").value
|
lastVersion.nextMinor().withSuffix("eap.1").value
|
||||||
|
@@ -14,7 +14,7 @@ fun main(args: Array<String>) {
|
|||||||
val releaseType = args[1]
|
val releaseType = args[1]
|
||||||
println("Working directory: $projectDir")
|
println("Working directory: $projectDir")
|
||||||
println("Release type: $releaseType")
|
println("Release type: $releaseType")
|
||||||
val (lastVersion, _) = getVersion(projectDir, onlyStable = true)
|
val (lastVersion, _) = getVersion(projectDir, ReleaseType.ONLY_STABLE)
|
||||||
|
|
||||||
val nextVersion = when (releaseType) {
|
val nextVersion = when (releaseType) {
|
||||||
"major" -> lastVersion.nextMajor()
|
"major" -> lastVersion.nextMajor()
|
||||||
|
@@ -8,28 +8,45 @@
|
|||||||
|
|
||||||
package scripts.release
|
package scripts.release
|
||||||
|
|
||||||
|
import com.vdurmont.semver4j.Semver
|
||||||
import java.time.LocalDate
|
import java.time.LocalDate
|
||||||
import java.time.format.DateTimeFormatter
|
import java.time.format.DateTimeFormatter
|
||||||
import kotlin.io.path.Path
|
import kotlin.io.path.Path
|
||||||
import kotlin.io.path.readText
|
import kotlin.io.path.readText
|
||||||
import kotlin.io.path.writeText
|
import kotlin.io.path.writeText
|
||||||
|
|
||||||
|
private const val toBeReleased = "## To Be Released"
|
||||||
|
|
||||||
fun main(args: Array<String>) {
|
fun main(args: Array<String>) {
|
||||||
println("Start updating unreleased section")
|
println("Start updating unreleased section")
|
||||||
val (newVersion, rootDir, releaseType) = readArgs(args)
|
val (newVersion, rootDir, releaseType) = readArgs(args)
|
||||||
|
|
||||||
checkReleaseType(releaseType)
|
checkReleaseType(releaseType)
|
||||||
|
|
||||||
if (releaseType == "patch") {
|
|
||||||
println("Skip updating the changelog because release type is 'patch'")
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
val currentDate = LocalDate.now().format(DateTimeFormatter.ISO_LOCAL_DATE)
|
val currentDate = LocalDate.now().format(DateTimeFormatter.ISO_LOCAL_DATE)
|
||||||
val newItem = "## $newVersion, $currentDate"
|
val newItem = "## $newVersion, $currentDate"
|
||||||
|
|
||||||
val changelogPath = Path("$rootDir/CHANGES.md")
|
val changelogPath = Path("$rootDir/CHANGES.md")
|
||||||
val changelog = changelogPath.readText()
|
val changelog = changelogPath.readText()
|
||||||
val newChangelog = changelog.replace("## To Be Released", newItem)
|
val newChangelog = if (releaseType == "patch") {
|
||||||
|
val decreasedVersion = Semver(newVersion).withIncPatch(-1)
|
||||||
|
val firstEntry = changelog.indexOf("## $decreasedVersion")
|
||||||
|
if (firstEntry != -1) {
|
||||||
|
val newLog = StringBuilder(changelog)
|
||||||
|
newLog.insert(firstEntry, newItem + "\n")
|
||||||
|
newLog.toString()
|
||||||
|
} else {
|
||||||
|
changelog
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (toBeReleased in changelog) {
|
||||||
|
changelog.replace(toBeReleased, newItem)
|
||||||
|
} else {
|
||||||
|
val firstEntry = changelog.indexOf("##")
|
||||||
|
val newLog = StringBuilder(changelog)
|
||||||
|
newLog.insert(firstEntry, newItem + "\n")
|
||||||
|
newLog.toString()
|
||||||
|
}
|
||||||
|
}
|
||||||
changelogPath.writeText(newChangelog)
|
changelogPath.writeText(newChangelog)
|
||||||
}
|
}
|
@@ -13,21 +13,20 @@ fun main(args: Array<String>) {
|
|||||||
|
|
||||||
checkReleaseType(releaseType)
|
checkReleaseType(releaseType)
|
||||||
|
|
||||||
if (releaseType == "patch") {
|
|
||||||
println("Skip committing changes because release type is 'patch'")
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
withGit(rootDir) { git ->
|
withGit(rootDir) { git ->
|
||||||
git
|
if (git.diff().call().isNotEmpty()) {
|
||||||
.commit()
|
git
|
||||||
.setAll(true)
|
.commit()
|
||||||
.setAuthor("IdeaVim Bot", "maintainers@ideavim.dev")
|
.setAll(true)
|
||||||
.setMessage("Preparation to $newVersion release")
|
.setAuthor("IdeaVim Bot", "maintainers@ideavim.dev")
|
||||||
.setSign(false)
|
.setMessage("Preparation to $newVersion release")
|
||||||
.call()
|
.setSign(false)
|
||||||
|
.call()
|
||||||
|
|
||||||
val lastGitMessage = git.log().call().first().shortMessage
|
val lastGitMessage = git.log().call().first().shortMessage
|
||||||
println("Changes committed. Last gitlog message: $lastGitMessage")
|
println("Changes committed. Last gitlog message: $lastGitMessage")
|
||||||
|
} else {
|
||||||
|
println("No Changes")
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -1,38 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright 2003-2023 The IdeaVim authors
|
|
||||||
*
|
|
||||||
* Use of this source code is governed by an MIT-style
|
|
||||||
* license that can be found in the LICENSE.txt file or at
|
|
||||||
* https://opensource.org/licenses/MIT.
|
|
||||||
*/
|
|
||||||
|
|
||||||
package scripts.release
|
|
||||||
|
|
||||||
fun main(args: Array<String>) {
|
|
||||||
val (_, rootDir, releaseType) = readArgs(args)
|
|
||||||
|
|
||||||
checkReleaseType(releaseType)
|
|
||||||
|
|
||||||
checkBranch(rootDir, releaseType)
|
|
||||||
|
|
||||||
if (releaseType == "patch") {
|
|
||||||
println("Skip release branch reset because release type is 'patch'")
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
withGit(rootDir) { git ->
|
|
||||||
val currentCommit = git.log().setMaxCount(1).call().first()
|
|
||||||
println("Current commit id: ${currentCommit.id.name}")
|
|
||||||
|
|
||||||
git.checkoutBranch("release")
|
|
||||||
println("Checked out release branch")
|
|
||||||
|
|
||||||
git.reset()
|
|
||||||
.setRef(currentCommit.id.name)
|
|
||||||
.call()
|
|
||||||
println("release branch reset")
|
|
||||||
|
|
||||||
git.checkoutBranch("master")
|
|
||||||
println("Checked out master branch")
|
|
||||||
}
|
|
||||||
}
|
|
@@ -15,8 +15,7 @@ fun main(args: Array<String>) {
|
|||||||
|
|
||||||
withGit(rootDir) { git ->
|
withGit(rootDir) { git ->
|
||||||
val branchName = when (releaseType) {
|
val branchName = when (releaseType) {
|
||||||
"major", "minor" -> "master"
|
"major", "minor", "patch" -> "release"
|
||||||
"patch" -> "release"
|
|
||||||
else -> error("Unsupported release type: $releaseType")
|
else -> error("Unsupported release type: $releaseType")
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -58,7 +58,13 @@ internal fun checkBranch(rootDir: String, releaseType: String) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
internal fun getVersion(projectDir: String, onlyStable: Boolean): Pair<Semver, ObjectId> {
|
enum class ReleaseType {
|
||||||
|
ANY,
|
||||||
|
ONLY_STABLE,
|
||||||
|
STABLE_NO_PATCH, // Version that ends on 0. Like 2.5.0
|
||||||
|
}
|
||||||
|
|
||||||
|
internal fun getVersion(projectDir: String, releaseType: ReleaseType): Pair<Semver, ObjectId> {
|
||||||
val repository = RepositoryBuilder().setGitDir(File("$projectDir/.git")).build()
|
val repository = RepositoryBuilder().setGitDir(File("$projectDir/.git")).build()
|
||||||
val git = Git(repository)
|
val git = Git(repository)
|
||||||
println(git.log().call().first())
|
println(git.log().call().first())
|
||||||
@@ -75,10 +81,10 @@ internal fun getVersion(projectDir: String, onlyStable: Boolean): Pair<Semver, O
|
|||||||
}
|
}
|
||||||
.sortedBy { it.first }
|
.sortedBy { it.first }
|
||||||
|
|
||||||
val version = if (onlyStable) {
|
val version = when (releaseType) {
|
||||||
versions.last { it.first.isStable }
|
ReleaseType.ANY -> versions.last()
|
||||||
} else {
|
ReleaseType.ONLY_STABLE -> versions.last { it.first.isStable }
|
||||||
versions.last()
|
ReleaseType.STABLE_NO_PATCH -> versions.last { it.first.isStable && it.first.patch == 0 }
|
||||||
}
|
}
|
||||||
|
|
||||||
return version
|
return version
|
||||||
|
@@ -0,0 +1,36 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2003-2024 The IdeaVim authors
|
||||||
|
*
|
||||||
|
* Use of this source code is governed by an MIT-style
|
||||||
|
* license that can be found in the LICENSE.txt file or at
|
||||||
|
* https://opensource.org/licenses/MIT.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package scripts.releaseEap
|
||||||
|
|
||||||
|
import kotlinx.coroutines.runBlocking
|
||||||
|
import scripts.addComment
|
||||||
|
import scripts.getYoutrackTicketsByQuery
|
||||||
|
import scripts.release.readArgs
|
||||||
|
import scripts.releasedInEapTagId
|
||||||
|
import scripts.setTag
|
||||||
|
|
||||||
|
fun main(args: Array<String>) {
|
||||||
|
runBlocking {
|
||||||
|
val (newVersion, _, _) = readArgs(args)
|
||||||
|
|
||||||
|
// Search for Ready to release, but without "IdeaVim Released In EAP" tag
|
||||||
|
val ticketsToUpdate =
|
||||||
|
getYoutrackTicketsByQuery("%23%7BReady%20To%20Release%7D%20tag:%20-%7BIdeaVim%20Released%20In%20EAP%7D%20")
|
||||||
|
println("Have to update the following tickets: $ticketsToUpdate")
|
||||||
|
|
||||||
|
ticketsToUpdate.forEach { ticketId ->
|
||||||
|
setTag(ticketId, releasedInEapTagId)
|
||||||
|
addComment(
|
||||||
|
ticketId, """
|
||||||
|
The fix is available in the IdeaVim $newVersion. See https://jb.gg/ideavim-eap for the instructions on how to get EAP builds as updates within the IDE. You can also wait till the next stable release with this fix, you’ll get it automatically.
|
||||||
|
""".trimIndent()
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@@ -11,6 +11,8 @@ package scripts
|
|||||||
import io.ktor.client.call.*
|
import io.ktor.client.call.*
|
||||||
import io.ktor.client.request.*
|
import io.ktor.client.request.*
|
||||||
import io.ktor.http.*
|
import io.ktor.http.*
|
||||||
|
import kotlinx.coroutines.runBlocking
|
||||||
|
import kotlinx.serialization.json.JsonArray
|
||||||
import kotlinx.serialization.json.JsonObject
|
import kotlinx.serialization.json.JsonObject
|
||||||
import kotlinx.serialization.json.addJsonObject
|
import kotlinx.serialization.json.addJsonObject
|
||||||
import kotlinx.serialization.json.buildJsonObject
|
import kotlinx.serialization.json.buildJsonObject
|
||||||
@@ -21,6 +23,10 @@ import kotlinx.serialization.json.put
|
|||||||
import kotlinx.serialization.json.putJsonArray
|
import kotlinx.serialization.json.putJsonArray
|
||||||
import kotlinx.serialization.json.putJsonObject
|
import kotlinx.serialization.json.putJsonObject
|
||||||
|
|
||||||
|
|
||||||
|
// YouTrack tag "IdeaVim Released In EAP"
|
||||||
|
const val releasedInEapTagId = "68-385032"
|
||||||
|
|
||||||
suspend fun setYoutrackStatus(tickets: Collection<String>, status: String) {
|
suspend fun setYoutrackStatus(tickets: Collection<String>, status: String) {
|
||||||
val client = httpClient()
|
val client = httpClient()
|
||||||
|
|
||||||
@@ -59,3 +65,59 @@ suspend fun setYoutrackStatus(tickets: Collection<String>, status: String) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fun getYoutrackTicketsByQuery(query: String): Set<String> {
|
||||||
|
val client = httpClient()
|
||||||
|
|
||||||
|
return runBlocking {
|
||||||
|
val response = client.get("https://youtrack.jetbrains.com/api/issues/?fields=idReadable&query=project:VIM+$query")
|
||||||
|
response.body<JsonArray>().mapTo(HashSet()) { it.jsonObject.getValue("idReadable").jsonPrimitive.content }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 68-385032
|
||||||
|
* [issueHumanId] is like VIM-123
|
||||||
|
* [tagId] is like "145-23"
|
||||||
|
*/
|
||||||
|
suspend fun setTag(issueHumanId: String, tagId: String) {
|
||||||
|
val client = httpClient()
|
||||||
|
|
||||||
|
println("Try to add tag $tagId to $issueHumanId")
|
||||||
|
val response =
|
||||||
|
// I've updated default url in client, so this may be broken now
|
||||||
|
client.post("https://youtrack.jetbrains.com/api/issues/$issueHumanId/tags?fields=customFields(id,name,value(id,name))") {
|
||||||
|
contentType(ContentType.Application.Json)
|
||||||
|
accept(ContentType.Application.Json)
|
||||||
|
val request = buildJsonObject {
|
||||||
|
put("id", tagId)
|
||||||
|
}
|
||||||
|
setBody(request)
|
||||||
|
}
|
||||||
|
println(response)
|
||||||
|
println(response.body<String>())
|
||||||
|
if (!response.status.isSuccess()) {
|
||||||
|
error("Request failed. $issueHumanId, ${response.body<String>()}")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
suspend fun addComment(issueHumanId: String, text: String) {
|
||||||
|
val client = httpClient()
|
||||||
|
|
||||||
|
println("Try to add comment to $issueHumanId")
|
||||||
|
val response =
|
||||||
|
// I've updated default url in client, so this may be broken now
|
||||||
|
client.post("https://youtrack.jetbrains.com/api/issues/$issueHumanId/comments?fields=customFields(id,name,value(id,name))") {
|
||||||
|
contentType(ContentType.Application.Json)
|
||||||
|
accept(ContentType.Application.Json)
|
||||||
|
val request = buildJsonObject {
|
||||||
|
put("text", text)
|
||||||
|
}
|
||||||
|
setBody(request)
|
||||||
|
}
|
||||||
|
println(response)
|
||||||
|
println(response.body<String>())
|
||||||
|
if (!response.status.isSuccess()) {
|
||||||
|
error("Request failed. $issueHumanId, ${response.body<String>()}")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@@ -12,4 +12,5 @@ rootProject.name = 'IdeaVIM'
|
|||||||
include 'vim-engine'
|
include 'vim-engine'
|
||||||
include 'scripts'
|
include 'scripts'
|
||||||
include 'annotation-processors'
|
include 'annotation-processors'
|
||||||
|
include 'tests:java-tests'
|
||||||
|
|
||||||
|
@@ -11,7 +11,7 @@ package com.maddyhome.idea.vim
|
|||||||
import com.intellij.openapi.fileEditor.ex.FileEditorManagerEx
|
import com.intellij.openapi.fileEditor.ex.FileEditorManagerEx
|
||||||
import com.intellij.openapi.project.Project
|
import com.intellij.openapi.project.Project
|
||||||
import com.intellij.openapi.project.ProjectManagerListener
|
import com.intellij.openapi.project.ProjectManagerListener
|
||||||
import com.intellij.openapi.startup.StartupActivity
|
import com.intellij.openapi.startup.ProjectActivity
|
||||||
import com.maddyhome.idea.vim.api.injector
|
import com.maddyhome.idea.vim.api.injector
|
||||||
import com.maddyhome.idea.vim.helper.EditorHelper
|
import com.maddyhome.idea.vim.helper.EditorHelper
|
||||||
import com.maddyhome.idea.vim.helper.localEditors
|
import com.maddyhome.idea.vim.helper.localEditors
|
||||||
@@ -20,16 +20,11 @@ import com.maddyhome.idea.vim.newapi.globalIjOptions
|
|||||||
/**
|
/**
|
||||||
* @author Alex Plate
|
* @author Alex Plate
|
||||||
*/
|
*/
|
||||||
// This service should be migrated to ProjectActivity. But we should cariful because simple replacement
|
internal class PluginStartup : ProjectActivity/*, LightEditCompatible*/ {
|
||||||
// leads to deadlock in tests. I'm not sure about the exact reasons, but "invokeAndWait" inside "initialize" function
|
|
||||||
// causes this deadlock. Good new: it's easy reproducible in tests.
|
|
||||||
// Previous migration: fc7efd5484a13b40ba9bf86a1d5429e215d973f3
|
|
||||||
// Revert: 24dd84b31cffb99eb6114524859a46d02717d33f
|
|
||||||
internal class PluginStartup : StartupActivity.DumbAware/*, LightEditCompatible*/ {
|
|
||||||
|
|
||||||
private var firstInitializationOccurred = false
|
private var firstInitializationOccurred = false
|
||||||
|
|
||||||
override fun runActivity(project: Project) {
|
override suspend fun execute(project: Project) {
|
||||||
if (firstInitializationOccurred) return
|
if (firstInitializationOccurred) return
|
||||||
firstInitializationOccurred = true
|
firstInitializationOccurred = true
|
||||||
|
|
||||||
|
@@ -7,56 +7,27 @@
|
|||||||
*/
|
*/
|
||||||
package com.maddyhome.idea.vim
|
package com.maddyhome.idea.vim
|
||||||
|
|
||||||
import com.intellij.openapi.application.ApplicationManager
|
|
||||||
import com.intellij.openapi.extensions.ExtensionPointName
|
|
||||||
import com.maddyhome.idea.vim.action.EngineCommandProvider
|
import com.maddyhome.idea.vim.action.EngineCommandProvider
|
||||||
import com.maddyhome.idea.vim.action.IntellijCommandProvider
|
import com.maddyhome.idea.vim.action.IntellijCommandProvider
|
||||||
import com.maddyhome.idea.vim.api.injector
|
|
||||||
import com.maddyhome.idea.vim.handler.ActionBeanClass
|
|
||||||
import com.maddyhome.idea.vim.handler.EditorActionHandlerBase
|
import com.maddyhome.idea.vim.handler.EditorActionHandlerBase
|
||||||
import com.maddyhome.idea.vim.key.MappingOwner
|
import com.maddyhome.idea.vim.key.MappingOwner
|
||||||
import com.maddyhome.idea.vim.newapi.IjVimActionsInitiator
|
|
||||||
import com.maddyhome.idea.vim.newapi.globalIjOptions
|
|
||||||
import java.awt.event.KeyEvent
|
import java.awt.event.KeyEvent
|
||||||
import javax.swing.KeyStroke
|
import javax.swing.KeyStroke
|
||||||
|
|
||||||
public object RegisterActions {
|
public object RegisterActions {
|
||||||
@Deprecated("Please use @CommandOrMotion annotation instead")
|
|
||||||
internal val VIM_ACTIONS_EP: ExtensionPointName<ActionBeanClass> = ExtensionPointName.create("IdeaVIM.vimAction")
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Register all the key/action mappings for the plugin.
|
* Register all the key/action mappings for the plugin.
|
||||||
*/
|
*/
|
||||||
@JvmStatic
|
@JvmStatic
|
||||||
public fun registerActions() {
|
public fun registerActions() {
|
||||||
registerVimCommandActions()
|
registerVimCommandActions()
|
||||||
if (!injector.globalIjOptions().commandOrMotionAnnotation) {
|
registerEmptyShortcuts() // todo most likely it is not needed
|
||||||
registerEmptyShortcuts()
|
|
||||||
registerEpListener()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@Deprecated("Moving to annotations approach instead of xml")
|
|
||||||
private fun registerEpListener() {
|
|
||||||
// IdeaVim doesn't support contribution to VIM_ACTIONS_EP extension point, so technically we can skip this update,
|
|
||||||
// but let's support dynamic plugins in a more classic way and reload actions on every EP change.
|
|
||||||
VIM_ACTIONS_EP.addChangeListener({
|
|
||||||
unregisterActions()
|
|
||||||
registerActions()
|
|
||||||
}, VimPlugin.getInstance())
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public fun findAction(id: String): EditorActionHandlerBase? {
|
public fun findAction(id: String): EditorActionHandlerBase? {
|
||||||
if (injector.globalIjOptions().commandOrMotionAnnotation) {
|
val commandBean = EngineCommandProvider.getCommands().firstOrNull { it.actionId == id }
|
||||||
val commandBean = EngineCommandProvider.getCommands().firstOrNull { it.actionId == id }
|
?: IntellijCommandProvider.getCommands().firstOrNull { it.actionId == id } ?: return null
|
||||||
?: IntellijCommandProvider.getCommands().firstOrNull { it.actionId == id } ?: return null
|
return commandBean.instance
|
||||||
return commandBean.instance
|
|
||||||
} else {
|
|
||||||
return VIM_ACTIONS_EP.getExtensionList(ApplicationManager.getApplication()).stream()
|
|
||||||
.filter { vimActionBean: ActionBeanClass -> vimActionBean.actionId == id }
|
|
||||||
.findFirst().map { obj: ActionBeanClass -> obj.instance }
|
|
||||||
.orElse(null)
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public fun findActionOrDie(id: String): EditorActionHandlerBase {
|
public fun findActionOrDie(id: String): EditorActionHandlerBase {
|
||||||
@@ -71,24 +42,10 @@ public object RegisterActions {
|
|||||||
|
|
||||||
private fun registerVimCommandActions() {
|
private fun registerVimCommandActions() {
|
||||||
val parser = VimPlugin.getKey()
|
val parser = VimPlugin.getKey()
|
||||||
if (injector.globalIjOptions().commandOrMotionAnnotation) {
|
EngineCommandProvider.getCommands().forEach { parser.registerCommandAction(it) }
|
||||||
EngineCommandProvider.getCommands().forEach { parser.registerCommandAction(it) }
|
IntellijCommandProvider.getCommands().forEach { parser.registerCommandAction(it) }
|
||||||
IntellijCommandProvider.getCommands().forEach { parser.registerCommandAction(it) }
|
|
||||||
} else {
|
|
||||||
VIM_ACTIONS_EP.getExtensionList(ApplicationManager.getApplication()).stream().map { bean: ActionBeanClass? ->
|
|
||||||
IjVimActionsInitiator(
|
|
||||||
bean!!
|
|
||||||
)
|
|
||||||
}
|
|
||||||
.forEach { actionHolder: IjVimActionsInitiator? ->
|
|
||||||
parser.registerCommandAction(
|
|
||||||
actionHolder!!
|
|
||||||
)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// todo do we really need this?
|
|
||||||
private fun registerEmptyShortcuts() {
|
private fun registerEmptyShortcuts() {
|
||||||
val parser = VimPlugin.getKey()
|
val parser = VimPlugin.getKey()
|
||||||
|
|
||||||
|
@@ -39,11 +39,9 @@ import com.maddyhome.idea.vim.listener.VimListenerManager;
|
|||||||
import com.maddyhome.idea.vim.newapi.IjVimInjector;
|
import com.maddyhome.idea.vim.newapi.IjVimInjector;
|
||||||
import com.maddyhome.idea.vim.ui.StatusBarIconFactory;
|
import com.maddyhome.idea.vim.ui.StatusBarIconFactory;
|
||||||
import com.maddyhome.idea.vim.ui.ex.ExEntryPanel;
|
import com.maddyhome.idea.vim.ui.ex.ExEntryPanel;
|
||||||
import com.maddyhome.idea.vim.vimscript.services.OptionService;
|
|
||||||
import com.maddyhome.idea.vim.vimscript.services.VariableService;
|
import com.maddyhome.idea.vim.vimscript.services.VariableService;
|
||||||
import com.maddyhome.idea.vim.yank.YankGroupBase;
|
import com.maddyhome.idea.vim.yank.YankGroupBase;
|
||||||
import org.jdom.Element;
|
import org.jdom.Element;
|
||||||
import org.jetbrains.annotations.ApiStatus;
|
|
||||||
import org.jetbrains.annotations.Nls;
|
import org.jetbrains.annotations.Nls;
|
||||||
import org.jetbrains.annotations.NotNull;
|
import org.jetbrains.annotations.NotNull;
|
||||||
import org.jetbrains.annotations.Nullable;
|
import org.jetbrains.annotations.Nullable;
|
||||||
@@ -117,12 +115,6 @@ public class VimPlugin implements PersistentStateComponent<Element>, Disposable
|
|||||||
return ApplicationManager.getApplication().getService(CommandGroup.class);
|
return ApplicationManager.getApplication().getService(CommandGroup.class);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Deprecated // "Please use `injector.markService` instead"
|
|
||||||
@ApiStatus.ScheduledForRemoval(inVersion = "2.3")
|
|
||||||
public static @NotNull MarkGroup getMark() {
|
|
||||||
return ApplicationManager.getApplication().getService(MarkGroup.class);
|
|
||||||
}
|
|
||||||
|
|
||||||
public static @NotNull RegisterGroup getRegister() {
|
public static @NotNull RegisterGroup getRegister() {
|
||||||
return ((RegisterGroup)VimInjectorKt.getInjector().getRegisterGroup());
|
return ((RegisterGroup)VimInjectorKt.getInjector().getRegisterGroup());
|
||||||
}
|
}
|
||||||
@@ -195,13 +187,6 @@ public class VimPlugin implements PersistentStateComponent<Element>, Disposable
|
|||||||
return VimInjectorKt.getInjector().getOptionGroup();
|
return VimInjectorKt.getInjector().getOptionGroup();
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Deprecated: Use getOptionGroup */
|
|
||||||
@Deprecated
|
|
||||||
// Used by which-key 0.8.0, IdeaVimExtension 1.6.5 + 1.6.8
|
|
||||||
public static @NotNull OptionService getOptionService() {
|
|
||||||
return VimInjectorKt.getInjector().getOptionService();
|
|
||||||
}
|
|
||||||
|
|
||||||
private static @NotNull NotificationService getNotifications() {
|
private static @NotNull NotificationService getNotifications() {
|
||||||
return getNotifications(null);
|
return getNotifications(null);
|
||||||
}
|
}
|
||||||
@@ -219,6 +204,10 @@ public class VimPlugin implements PersistentStateComponent<Element>, Disposable
|
|||||||
return getInstance().enabled;
|
return getInstance().enabled;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public static boolean isNotEnabled() {
|
||||||
|
return !isEnabled();
|
||||||
|
}
|
||||||
|
|
||||||
public static void setEnabled(final boolean enabled) {
|
public static void setEnabled(final boolean enabled) {
|
||||||
if (isEnabled() == enabled) return;
|
if (isEnabled() == enabled) return;
|
||||||
|
|
||||||
@@ -232,7 +221,13 @@ public class VimPlugin implements PersistentStateComponent<Element>, Disposable
|
|||||||
getInstance().turnOnPlugin();
|
getInstance().turnOnPlugin();
|
||||||
}
|
}
|
||||||
|
|
||||||
StatusBarIconFactory.Companion.updateIcon();
|
if (enabled) {
|
||||||
|
VimInjectorKt.getInjector().getListenersNotifier().notifyPluginTurnedOn();
|
||||||
|
} else {
|
||||||
|
VimInjectorKt.getInjector().getListenersNotifier().notifyPluginTurnedOff();
|
||||||
|
}
|
||||||
|
|
||||||
|
StatusBarIconFactory.Util.INSTANCE.updateIcon();
|
||||||
}
|
}
|
||||||
|
|
||||||
public static String getMessage() {
|
public static String getMessage() {
|
||||||
@@ -264,7 +259,8 @@ public class VimPlugin implements PersistentStateComponent<Element>, Disposable
|
|||||||
if (enabled) {
|
if (enabled) {
|
||||||
Application application = ApplicationManager.getApplication();
|
Application application = ApplicationManager.getApplication();
|
||||||
if (application.isUnitTestMode()) {
|
if (application.isUnitTestMode()) {
|
||||||
application.invokeAndWait(this::turnOnPlugin);
|
turnOnPlugin();
|
||||||
|
//application.invokeAndWait(this::turnOnPlugin);
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
application.invokeLater(this::turnOnPlugin);
|
application.invokeLater(this::turnOnPlugin);
|
||||||
|
@@ -28,8 +28,11 @@ import javax.swing.KeyStroke
|
|||||||
* Accepts all regular keystrokes and passes them on to the Vim key handler.
|
* Accepts all regular keystrokes and passes them on to the Vim key handler.
|
||||||
*
|
*
|
||||||
* IDE shortcut keys used by Vim commands are handled by [com.maddyhome.idea.vim.action.VimShortcutKeyAction].
|
* IDE shortcut keys used by Vim commands are handled by [com.maddyhome.idea.vim.action.VimShortcutKeyAction].
|
||||||
|
*
|
||||||
|
* This class is used in Which-Key plugin, so don't make it internal. Generally, we should provide a proper
|
||||||
|
* way to get ideavim keys for this plugin. See VIM-3085
|
||||||
*/
|
*/
|
||||||
internal class VimTypedActionHandler(origHandler: TypedActionHandler) : TypedActionHandlerEx {
|
public class VimTypedActionHandler(origHandler: TypedActionHandler) : TypedActionHandlerEx {
|
||||||
private val handler = KeyHandler.getInstance()
|
private val handler = KeyHandler.getInstance()
|
||||||
private val traceTime = injector.globalOptions().ideatracetime
|
private val traceTime = injector.globalOptions().ideatracetime
|
||||||
|
|
||||||
@@ -86,7 +89,7 @@ internal class VimTypedActionHandler(origHandler: TypedActionHandler) : TypedAct
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
companion object {
|
internal companion object {
|
||||||
private val LOG = logger<VimTypedActionHandler>()
|
private val LOG = logger<VimTypedActionHandler>()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -1,74 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright 2003-2023 The IdeaVim authors
|
|
||||||
*
|
|
||||||
* Use of this source code is governed by an MIT-style
|
|
||||||
* license that can be found in the LICENSE.txt file or at
|
|
||||||
* https://opensource.org/licenses/MIT.
|
|
||||||
*/
|
|
||||||
package com.maddyhome.idea.vim.action
|
|
||||||
|
|
||||||
import com.intellij.codeInsight.hint.HintManagerImpl
|
|
||||||
import com.intellij.openapi.actionSystem.ActionManager
|
|
||||||
import com.intellij.openapi.actionSystem.ActionUpdateThread
|
|
||||||
import com.intellij.openapi.actionSystem.AnAction
|
|
||||||
import com.intellij.openapi.actionSystem.AnActionEvent
|
|
||||||
import com.intellij.openapi.actionSystem.CommonDataKeys
|
|
||||||
import com.intellij.openapi.actionSystem.PerformWithDocumentsCommitted
|
|
||||||
import com.intellij.openapi.actionSystem.PopupAction
|
|
||||||
import com.intellij.openapi.actionSystem.impl.ActionConfigurationCustomizer
|
|
||||||
import com.intellij.openapi.editor.Editor
|
|
||||||
import com.intellij.openapi.editor.EditorMouseHoverPopupManager
|
|
||||||
import com.intellij.openapi.editor.event.EditorMouseEvent
|
|
||||||
import com.intellij.openapi.editor.event.EditorMouseEventArea
|
|
||||||
import com.intellij.openapi.project.DumbAware
|
|
||||||
import java.awt.event.MouseEvent
|
|
||||||
|
|
||||||
// [VERSION UPDATE] 233+ Remove class
|
|
||||||
// The ShowHoverInfo action is built into the platform (using a nicer EditorMouseHoverPopupManager API)
|
|
||||||
public class VimActionConfigurationCustomizer : ActionConfigurationCustomizer {
|
|
||||||
public override fun customize(actionManager: ActionManager) {
|
|
||||||
// If the ShowHoverInfo action doesn't exist in the platform, add our own implementation
|
|
||||||
if (actionManager.getAction("ShowHoverInfo") == null) {
|
|
||||||
actionManager.registerAction("ShowHoverInfo", VimShowHoverInfoAction())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private class VimShowHoverInfoAction : AnAction(), HintManagerImpl.ActionToIgnore, PopupAction, DumbAware,
|
|
||||||
PerformWithDocumentsCommitted {
|
|
||||||
override fun getActionUpdateThread(): ActionUpdateThread = ActionUpdateThread.BGT
|
|
||||||
|
|
||||||
override fun update(e: AnActionEvent) {
|
|
||||||
val dataContext = e.dataContext
|
|
||||||
val editor = CommonDataKeys.EDITOR.getData(dataContext)
|
|
||||||
if (editor == null) {
|
|
||||||
e.presentation.isEnabledAndVisible = false
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
override fun actionPerformed(e: AnActionEvent) {
|
|
||||||
val editor = CommonDataKeys.EDITOR.getData(e.dataContext) ?: return
|
|
||||||
|
|
||||||
val editorMouseEvent = createFakeEditorMouseEvent(editor)
|
|
||||||
EditorMouseHoverPopupManager.getInstance().showInfoTooltip(editorMouseEvent)
|
|
||||||
}
|
|
||||||
|
|
||||||
private fun createFakeEditorMouseEvent(editor: Editor): EditorMouseEvent {
|
|
||||||
val xy = editor.offsetToXY(editor.caretModel.offset)
|
|
||||||
val mouseEvent =
|
|
||||||
MouseEvent(editor.component, MouseEvent.MOUSE_MOVED, System.currentTimeMillis(), 0, xy.x, xy.y, 0, false)
|
|
||||||
val editorMouseEvent = EditorMouseEvent(
|
|
||||||
editor,
|
|
||||||
mouseEvent,
|
|
||||||
EditorMouseEventArea.EDITING_AREA,
|
|
||||||
editor.caretModel.offset,
|
|
||||||
editor.caretModel.logicalPosition,
|
|
||||||
editor.caretModel.visualPosition,
|
|
||||||
true,
|
|
||||||
null,
|
|
||||||
null,
|
|
||||||
null
|
|
||||||
)
|
|
||||||
return editorMouseEvent
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
@@ -28,6 +28,7 @@ import com.maddyhome.idea.vim.api.globalOptions
|
|||||||
import com.maddyhome.idea.vim.api.injector
|
import com.maddyhome.idea.vim.api.injector
|
||||||
import com.maddyhome.idea.vim.group.IjOptionConstants
|
import com.maddyhome.idea.vim.group.IjOptionConstants
|
||||||
import com.maddyhome.idea.vim.group.IjOptions
|
import com.maddyhome.idea.vim.group.IjOptions
|
||||||
|
import com.maddyhome.idea.vim.handler.enableOctopus
|
||||||
import com.maddyhome.idea.vim.handler.isOctopusEnabled
|
import com.maddyhome.idea.vim.handler.isOctopusEnabled
|
||||||
import com.maddyhome.idea.vim.helper.EditorHelper
|
import com.maddyhome.idea.vim.helper.EditorHelper
|
||||||
import com.maddyhome.idea.vim.helper.HandlerInjector
|
import com.maddyhome.idea.vim.helper.HandlerInjector
|
||||||
@@ -54,9 +55,17 @@ import javax.swing.KeyStroke
|
|||||||
*
|
*
|
||||||
*
|
*
|
||||||
* These keys are not passed to [com.maddyhome.idea.vim.VimTypedActionHandler] and should be handled by actions.
|
* These keys are not passed to [com.maddyhome.idea.vim.VimTypedActionHandler] and should be handled by actions.
|
||||||
|
*
|
||||||
|
* This class is used in Which-Key plugin, so don't make it internal. Generally, we should provide a proper
|
||||||
|
* way to get ideavim keys for this plugin. See VIM-3085
|
||||||
*/
|
*/
|
||||||
internal class VimShortcutKeyAction : AnAction(), DumbAware/*, LightEditCompatible*/ {
|
public class VimShortcutKeyAction : AnAction(), DumbAware/*, LightEditCompatible*/ {
|
||||||
private val traceTime = injector.globalOptions().ideatracetime
|
private val traceTime: Boolean
|
||||||
|
get() {
|
||||||
|
// Make sure the injector is initialized
|
||||||
|
VimPlugin.getInstance()
|
||||||
|
return injector.globalOptions().ideatracetime
|
||||||
|
}
|
||||||
|
|
||||||
override fun actionPerformed(e: AnActionEvent) {
|
override fun actionPerformed(e: AnActionEvent) {
|
||||||
LOG.trace("Executing shortcut key action")
|
LOG.trace("Executing shortcut key action")
|
||||||
@@ -90,7 +99,7 @@ internal class VimShortcutKeyAction : AnAction(), DumbAware/*, LightEditCompatib
|
|||||||
|
|
||||||
// There is a chance that we can use BGT, but we call for isCell inside the update.
|
// There is a chance that we can use BGT, but we call for isCell inside the update.
|
||||||
// Not sure if can can use BGT with this call. Let's use EDT for now.
|
// Not sure if can can use BGT with this call. Let's use EDT for now.
|
||||||
override fun getActionUpdateThread() = ActionUpdateThread.EDT
|
override fun getActionUpdateThread(): ActionUpdateThread = ActionUpdateThread.EDT
|
||||||
|
|
||||||
override fun update(e: AnActionEvent) {
|
override fun update(e: AnActionEvent) {
|
||||||
val start = if (traceTime) System.currentTimeMillis() else null
|
val start = if (traceTime) System.currentTimeMillis() else null
|
||||||
@@ -105,14 +114,16 @@ internal class VimShortcutKeyAction : AnAction(), DumbAware/*, LightEditCompatib
|
|||||||
}
|
}
|
||||||
|
|
||||||
private fun isEnabled(e: AnActionEvent, keyStroke: KeyStroke?): ActionEnableStatus {
|
private fun isEnabled(e: AnActionEvent, keyStroke: KeyStroke?): ActionEnableStatus {
|
||||||
if (!VimPlugin.isEnabled()) return ActionEnableStatus.no("IdeaVim is disabled", LogLevel.DEBUG)
|
if (VimPlugin.isNotEnabled()) return ActionEnableStatus.no("IdeaVim is disabled", LogLevel.DEBUG)
|
||||||
val editor = getEditor(e)
|
val editor = getEditor(e)
|
||||||
if (editor != null && keyStroke != null) {
|
if (editor != null && keyStroke != null) {
|
||||||
if (isOctopusEnabled(keyStroke, editor)) {
|
if (enableOctopus) {
|
||||||
return ActionEnableStatus.no(
|
if (isOctopusEnabled(keyStroke, editor)) {
|
||||||
"Processing VimShortcutKeyAction for the key that is used in the octopus handler",
|
return ActionEnableStatus.no(
|
||||||
LogLevel.ERROR
|
"Processing VimShortcutKeyAction for the key that is used in the octopus handler",
|
||||||
)
|
LogLevel.ERROR
|
||||||
|
)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
if (editor.isIdeaVimDisabledHere) {
|
if (editor.isIdeaVimDisabledHere) {
|
||||||
return ActionEnableStatus.no("IdeaVim is disabled in this place", LogLevel.INFO)
|
return ActionEnableStatus.no("IdeaVim is disabled in this place", LogLevel.INFO)
|
||||||
@@ -224,9 +235,9 @@ internal class VimShortcutKeyAction : AnAction(), DumbAware/*, LightEditCompatib
|
|||||||
/**
|
/**
|
||||||
* getDefaultKeyStroke is needed for NEO layout keyboard VIM-987
|
* getDefaultKeyStroke is needed for NEO layout keyboard VIM-987
|
||||||
* but we should cache the value because on the second call (isEnabled -> actionPerformed)
|
* but we should cache the value because on the second call (isEnabled -> actionPerformed)
|
||||||
* the event is already consumed
|
* the event is already consumed and getDefaultKeyStroke returns null
|
||||||
*/
|
*/
|
||||||
private var keyStrokeCache: Pair<KeyEvent?, KeyStroke?> = null to null
|
private var keyStrokeCache: Pair<Long?, KeyStroke?> = null to null
|
||||||
|
|
||||||
private fun getKeyStroke(e: AnActionEvent): KeyStroke? {
|
private fun getKeyStroke(e: AnActionEvent): KeyStroke? {
|
||||||
val inputEvent = e.inputEvent
|
val inputEvent = e.inputEvent
|
||||||
@@ -234,9 +245,9 @@ internal class VimShortcutKeyAction : AnAction(), DumbAware/*, LightEditCompatib
|
|||||||
val defaultKeyStroke = KeyStrokeAdapter.getDefaultKeyStroke(inputEvent)
|
val defaultKeyStroke = KeyStrokeAdapter.getDefaultKeyStroke(inputEvent)
|
||||||
val strokeCache = keyStrokeCache
|
val strokeCache = keyStrokeCache
|
||||||
if (defaultKeyStroke != null) {
|
if (defaultKeyStroke != null) {
|
||||||
keyStrokeCache = inputEvent to defaultKeyStroke
|
keyStrokeCache = inputEvent.`when` to defaultKeyStroke
|
||||||
return defaultKeyStroke
|
return defaultKeyStroke
|
||||||
} else if (strokeCache.first === inputEvent) {
|
} else if (strokeCache.first == inputEvent.`when`) {
|
||||||
keyStrokeCache = null to null
|
keyStrokeCache = null to null
|
||||||
return strokeCache.second
|
return strokeCache.second
|
||||||
}
|
}
|
||||||
@@ -269,7 +280,7 @@ internal class VimShortcutKeyAction : AnAction(), DumbAware/*, LightEditCompatib
|
|||||||
.toSet()
|
.toSet()
|
||||||
}
|
}
|
||||||
|
|
||||||
companion object {
|
internal companion object {
|
||||||
@JvmField
|
@JvmField
|
||||||
val VIM_ONLY_EDITOR_KEYS: Set<KeyStroke> =
|
val VIM_ONLY_EDITOR_KEYS: Set<KeyStroke> =
|
||||||
ImmutableSet.builder<KeyStroke>().addAll(getKeyStrokes(KeyEvent.VK_ENTER, 0))
|
ImmutableSet.builder<KeyStroke>().addAll(getKeyStrokes(KeyEvent.VK_ENTER, 0))
|
||||||
|
@@ -18,9 +18,7 @@ import com.maddyhome.idea.vim.api.injector
|
|||||||
import com.maddyhome.idea.vim.api.setChangeMarks
|
import com.maddyhome.idea.vim.api.setChangeMarks
|
||||||
import com.maddyhome.idea.vim.command.Argument
|
import com.maddyhome.idea.vim.command.Argument
|
||||||
import com.maddyhome.idea.vim.command.Command
|
import com.maddyhome.idea.vim.command.Command
|
||||||
import com.maddyhome.idea.vim.command.CommandFlags
|
|
||||||
import com.maddyhome.idea.vim.command.OperatorArguments
|
import com.maddyhome.idea.vim.command.OperatorArguments
|
||||||
import com.maddyhome.idea.vim.state.mode.SelectionType
|
|
||||||
import com.maddyhome.idea.vim.common.TextRange
|
import com.maddyhome.idea.vim.common.TextRange
|
||||||
import com.maddyhome.idea.vim.common.argumentCaptured
|
import com.maddyhome.idea.vim.common.argumentCaptured
|
||||||
import com.maddyhome.idea.vim.group.MotionGroup
|
import com.maddyhome.idea.vim.group.MotionGroup
|
||||||
@@ -28,10 +26,9 @@ import com.maddyhome.idea.vim.group.visual.VimSelection
|
|||||||
import com.maddyhome.idea.vim.handler.VimActionHandler
|
import com.maddyhome.idea.vim.handler.VimActionHandler
|
||||||
import com.maddyhome.idea.vim.handler.VisualOperatorActionHandler
|
import com.maddyhome.idea.vim.handler.VisualOperatorActionHandler
|
||||||
import com.maddyhome.idea.vim.helper.MessageHelper
|
import com.maddyhome.idea.vim.helper.MessageHelper
|
||||||
import com.maddyhome.idea.vim.helper.enumSetOf
|
|
||||||
import com.maddyhome.idea.vim.helper.vimStateMachine
|
import com.maddyhome.idea.vim.helper.vimStateMachine
|
||||||
import com.maddyhome.idea.vim.newapi.ij
|
import com.maddyhome.idea.vim.newapi.ij
|
||||||
import java.util.*
|
import com.maddyhome.idea.vim.state.mode.SelectionType
|
||||||
|
|
||||||
// todo make it multicaret
|
// todo make it multicaret
|
||||||
private fun doOperatorAction(editor: VimEditor, context: ExecutionContext, textRange: TextRange, selectionType: SelectionType): Boolean {
|
private fun doOperatorAction(editor: VimEditor, context: ExecutionContext, textRange: TextRange, selectionType: SelectionType): Boolean {
|
||||||
@@ -104,8 +101,6 @@ internal class OperatorAction : VimActionHandler.SingleExecution() {
|
|||||||
internal class VisualOperatorAction : VisualOperatorActionHandler.ForEachCaret() {
|
internal class VisualOperatorAction : VisualOperatorActionHandler.ForEachCaret() {
|
||||||
override val type: Command.Type = Command.Type.OTHER_SELF_SYNCHRONIZED
|
override val type: Command.Type = Command.Type.OTHER_SELF_SYNCHRONIZED
|
||||||
|
|
||||||
override val flags: EnumSet<CommandFlags> = enumSetOf(CommandFlags.FLAG_EXIT_VISUAL)
|
|
||||||
|
|
||||||
override fun executeAction(
|
override fun executeAction(
|
||||||
editor: VimEditor,
|
editor: VimEditor,
|
||||||
caret: VimCaret,
|
caret: VimCaret,
|
||||||
|
@@ -14,13 +14,10 @@ import com.maddyhome.idea.vim.api.VimCaret
|
|||||||
import com.maddyhome.idea.vim.api.VimEditor
|
import com.maddyhome.idea.vim.api.VimEditor
|
||||||
import com.maddyhome.idea.vim.api.injector
|
import com.maddyhome.idea.vim.api.injector
|
||||||
import com.maddyhome.idea.vim.command.Command
|
import com.maddyhome.idea.vim.command.Command
|
||||||
import com.maddyhome.idea.vim.command.CommandFlags
|
|
||||||
import com.maddyhome.idea.vim.command.OperatorArguments
|
import com.maddyhome.idea.vim.command.OperatorArguments
|
||||||
import com.maddyhome.idea.vim.group.visual.VimSelection
|
import com.maddyhome.idea.vim.group.visual.VimSelection
|
||||||
import com.maddyhome.idea.vim.handler.VisualOperatorActionHandler
|
import com.maddyhome.idea.vim.handler.VisualOperatorActionHandler
|
||||||
import com.maddyhome.idea.vim.helper.enumSetOf
|
|
||||||
import com.maddyhome.idea.vim.newapi.ijOptions
|
import com.maddyhome.idea.vim.newapi.ijOptions
|
||||||
import java.util.*
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @author vlan
|
* @author vlan
|
||||||
@@ -29,8 +26,6 @@ import java.util.*
|
|||||||
public class DeleteJoinVisualLinesAction : VisualOperatorActionHandler.SingleExecution() {
|
public class DeleteJoinVisualLinesAction : VisualOperatorActionHandler.SingleExecution() {
|
||||||
override val type: Command.Type = Command.Type.DELETE
|
override val type: Command.Type = Command.Type.DELETE
|
||||||
|
|
||||||
override val flags: EnumSet<CommandFlags> = enumSetOf(CommandFlags.FLAG_EXIT_VISUAL)
|
|
||||||
|
|
||||||
override fun executeForAllCarets(
|
override fun executeForAllCarets(
|
||||||
editor: VimEditor,
|
editor: VimEditor,
|
||||||
context: ExecutionContext,
|
context: ExecutionContext,
|
||||||
|
@@ -14,13 +14,10 @@ import com.maddyhome.idea.vim.api.VimCaret
|
|||||||
import com.maddyhome.idea.vim.api.VimEditor
|
import com.maddyhome.idea.vim.api.VimEditor
|
||||||
import com.maddyhome.idea.vim.api.injector
|
import com.maddyhome.idea.vim.api.injector
|
||||||
import com.maddyhome.idea.vim.command.Command
|
import com.maddyhome.idea.vim.command.Command
|
||||||
import com.maddyhome.idea.vim.command.CommandFlags
|
|
||||||
import com.maddyhome.idea.vim.command.OperatorArguments
|
import com.maddyhome.idea.vim.command.OperatorArguments
|
||||||
import com.maddyhome.idea.vim.group.visual.VimSelection
|
import com.maddyhome.idea.vim.group.visual.VimSelection
|
||||||
import com.maddyhome.idea.vim.handler.VisualOperatorActionHandler
|
import com.maddyhome.idea.vim.handler.VisualOperatorActionHandler
|
||||||
import com.maddyhome.idea.vim.helper.enumSetOf
|
|
||||||
import com.maddyhome.idea.vim.newapi.ijOptions
|
import com.maddyhome.idea.vim.newapi.ijOptions
|
||||||
import java.util.*
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @author vlan
|
* @author vlan
|
||||||
@@ -29,8 +26,6 @@ import java.util.*
|
|||||||
public class DeleteJoinVisualLinesSpacesAction : VisualOperatorActionHandler.SingleExecution() {
|
public class DeleteJoinVisualLinesSpacesAction : VisualOperatorActionHandler.SingleExecution() {
|
||||||
override val type: Command.Type = Command.Type.DELETE
|
override val type: Command.Type = Command.Type.DELETE
|
||||||
|
|
||||||
override val flags: EnumSet<CommandFlags> = enumSetOf(CommandFlags.FLAG_EXIT_VISUAL)
|
|
||||||
|
|
||||||
override fun executeForAllCarets(
|
override fun executeForAllCarets(
|
||||||
editor: VimEditor,
|
editor: VimEditor,
|
||||||
context: ExecutionContext,
|
context: ExecutionContext,
|
||||||
|
@@ -8,10 +8,9 @@
|
|||||||
|
|
||||||
package com.maddyhome.idea.vim.action.editor
|
package com.maddyhome.idea.vim.action.editor
|
||||||
|
|
||||||
|
import com.intellij.openapi.actionSystem.IdeActions
|
||||||
import com.intellij.vim.annotations.CommandOrMotion
|
import com.intellij.vim.annotations.CommandOrMotion
|
||||||
import com.intellij.vim.annotations.Mode
|
import com.intellij.vim.annotations.Mode
|
||||||
import com.intellij.openapi.actionSystem.IdeActions
|
|
||||||
import com.maddyhome.idea.vim.action.ComplicatedKeysAction
|
|
||||||
import com.maddyhome.idea.vim.api.ExecutionContext
|
import com.maddyhome.idea.vim.api.ExecutionContext
|
||||||
import com.maddyhome.idea.vim.api.VimEditor
|
import com.maddyhome.idea.vim.api.VimEditor
|
||||||
import com.maddyhome.idea.vim.api.injector
|
import com.maddyhome.idea.vim.api.injector
|
||||||
@@ -21,54 +20,32 @@ import com.maddyhome.idea.vim.command.OperatorArguments
|
|||||||
import com.maddyhome.idea.vim.handler.IdeActionHandler
|
import com.maddyhome.idea.vim.handler.IdeActionHandler
|
||||||
import com.maddyhome.idea.vim.handler.VimActionHandler
|
import com.maddyhome.idea.vim.handler.VimActionHandler
|
||||||
import com.maddyhome.idea.vim.helper.enumSetOf
|
import com.maddyhome.idea.vim.helper.enumSetOf
|
||||||
import java.awt.event.KeyEvent
|
|
||||||
import java.util.*
|
import java.util.*
|
||||||
import javax.swing.KeyStroke
|
|
||||||
|
|
||||||
@CommandOrMotion(keys = ["<C-H>", "<BS>"], modes = [Mode.INSERT])
|
@CommandOrMotion(keys = ["<C-H>", "<BS>"], modes = [Mode.INSERT])
|
||||||
internal class VimEditorBackSpace : IdeActionHandler(IdeActions.ACTION_EDITOR_BACKSPACE), ComplicatedKeysAction {
|
internal class VimEditorBackSpace : IdeActionHandler(IdeActions.ACTION_EDITOR_BACKSPACE) {
|
||||||
override val keyStrokesSet: Set<List<KeyStroke>> = setOf(
|
|
||||||
listOf(KeyStroke.getKeyStroke(KeyEvent.VK_H, KeyEvent.CTRL_DOWN_MASK)),
|
|
||||||
listOf(KeyStroke.getKeyStroke(KeyEvent.VK_BACK_SPACE, 0)),
|
|
||||||
)
|
|
||||||
override val type: Command.Type = Command.Type.DELETE
|
override val type: Command.Type = Command.Type.DELETE
|
||||||
}
|
}
|
||||||
|
|
||||||
@CommandOrMotion(keys = ["<Del>"], modes = [Mode.INSERT])
|
@CommandOrMotion(keys = ["<Del>"], modes = [Mode.INSERT])
|
||||||
internal class VimEditorDelete : IdeActionHandler(IdeActions.ACTION_EDITOR_DELETE), ComplicatedKeysAction {
|
internal class VimEditorDelete : IdeActionHandler(IdeActions.ACTION_EDITOR_DELETE) {
|
||||||
override val keyStrokesSet: Set<List<KeyStroke>> = setOf(
|
|
||||||
listOf(KeyStroke.getKeyStroke(KeyEvent.VK_DELETE, 0)),
|
|
||||||
)
|
|
||||||
override val type: Command.Type = Command.Type.DELETE
|
override val type: Command.Type = Command.Type.DELETE
|
||||||
override val flags: EnumSet<CommandFlags> = enumSetOf(CommandFlags.FLAG_SAVE_STROKE)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@CommandOrMotion(keys = ["<Down>", "<kDown>"], modes = [Mode.INSERT])
|
@CommandOrMotion(keys = ["<Down>", "<kDown>"], modes = [Mode.INSERT])
|
||||||
internal class VimEditorDown : IdeActionHandler(IdeActions.ACTION_EDITOR_MOVE_CARET_DOWN), ComplicatedKeysAction {
|
internal class VimEditorDown : IdeActionHandler(IdeActions.ACTION_EDITOR_MOVE_CARET_DOWN) {
|
||||||
override val keyStrokesSet: Set<List<KeyStroke>> = setOf(
|
|
||||||
listOf(KeyStroke.getKeyStroke(KeyEvent.VK_DOWN, 0)),
|
|
||||||
listOf(KeyStroke.getKeyStroke(KeyEvent.VK_KP_DOWN, 0)),
|
|
||||||
)
|
|
||||||
override val type: Command.Type = Command.Type.MOTION
|
override val type: Command.Type = Command.Type.MOTION
|
||||||
override val flags: EnumSet<CommandFlags> = enumSetOf(CommandFlags.FLAG_CLEAR_STROKES)
|
override val flags: EnumSet<CommandFlags> = enumSetOf(CommandFlags.FLAG_CLEAR_STROKES)
|
||||||
}
|
}
|
||||||
|
|
||||||
@CommandOrMotion(keys = ["<Tab>", "<C-I>"], modes = [Mode.INSERT])
|
@CommandOrMotion(keys = ["<Tab>", "<C-I>"], modes = [Mode.INSERT])
|
||||||
internal class VimEditorTab : IdeActionHandler(IdeActions.ACTION_EDITOR_TAB), ComplicatedKeysAction {
|
internal class VimEditorTab : IdeActionHandler(IdeActions.ACTION_EDITOR_TAB) {
|
||||||
override val keyStrokesSet: Set<List<KeyStroke>> = setOf(
|
|
||||||
listOf(KeyStroke.getKeyStroke(KeyEvent.VK_I, KeyEvent.CTRL_DOWN_MASK)),
|
|
||||||
listOf(KeyStroke.getKeyStroke(KeyEvent.VK_TAB, 0)),
|
|
||||||
)
|
|
||||||
override val type: Command.Type = Command.Type.INSERT
|
override val type: Command.Type = Command.Type.INSERT
|
||||||
override val flags: EnumSet<CommandFlags> = enumSetOf(CommandFlags.FLAG_SAVE_STROKE)
|
override val flags: EnumSet<CommandFlags> = enumSetOf(CommandFlags.FLAG_SAVE_STROKE)
|
||||||
}
|
}
|
||||||
|
|
||||||
@CommandOrMotion(keys = ["<Up>", "<kUp>"], modes = [Mode.INSERT])
|
@CommandOrMotion(keys = ["<Up>", "<kUp>"], modes = [Mode.INSERT])
|
||||||
internal class VimEditorUp : IdeActionHandler(IdeActions.ACTION_EDITOR_MOVE_CARET_UP), ComplicatedKeysAction {
|
internal class VimEditorUp : IdeActionHandler(IdeActions.ACTION_EDITOR_MOVE_CARET_UP) {
|
||||||
override val keyStrokesSet: Set<List<KeyStroke>> = setOf(
|
|
||||||
listOf(KeyStroke.getKeyStroke(KeyEvent.VK_UP, 0)),
|
|
||||||
listOf(KeyStroke.getKeyStroke(KeyEvent.VK_KP_UP, 0)),
|
|
||||||
)
|
|
||||||
override val type: Command.Type = Command.Type.MOTION
|
override val type: Command.Type = Command.Type.MOTION
|
||||||
override val flags: EnumSet<CommandFlags> = enumSetOf(CommandFlags.FLAG_CLEAR_STROKES)
|
override val flags: EnumSet<CommandFlags> = enumSetOf(CommandFlags.FLAG_CLEAR_STROKES)
|
||||||
}
|
}
|
||||||
|
@@ -10,7 +10,6 @@ package com.maddyhome.idea.vim.action.ex
|
|||||||
import com.intellij.vim.annotations.CommandOrMotion
|
import com.intellij.vim.annotations.CommandOrMotion
|
||||||
import com.intellij.vim.annotations.Mode
|
import com.intellij.vim.annotations.Mode
|
||||||
import com.maddyhome.idea.vim.VimPlugin
|
import com.maddyhome.idea.vim.VimPlugin
|
||||||
import com.maddyhome.idea.vim.action.ComplicatedKeysAction
|
|
||||||
import com.maddyhome.idea.vim.api.ExecutionContext
|
import com.maddyhome.idea.vim.api.ExecutionContext
|
||||||
import com.maddyhome.idea.vim.api.VimEditor
|
import com.maddyhome.idea.vim.api.VimEditor
|
||||||
import com.maddyhome.idea.vim.command.Command
|
import com.maddyhome.idea.vim.command.Command
|
||||||
@@ -18,7 +17,6 @@ import com.maddyhome.idea.vim.command.CommandFlags
|
|||||||
import com.maddyhome.idea.vim.command.OperatorArguments
|
import com.maddyhome.idea.vim.command.OperatorArguments
|
||||||
import com.maddyhome.idea.vim.handler.VimActionHandler
|
import com.maddyhome.idea.vim.handler.VimActionHandler
|
||||||
import java.util.*
|
import java.util.*
|
||||||
import javax.swing.KeyStroke
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Called by KeyHandler to process the contents of the ex entry panel
|
* Called by KeyHandler to process the contents of the ex entry panel
|
||||||
@@ -26,10 +24,7 @@ import javax.swing.KeyStroke
|
|||||||
* The mapping for this action means that the ex command is executed as a write action
|
* The mapping for this action means that the ex command is executed as a write action
|
||||||
*/
|
*/
|
||||||
@CommandOrMotion(keys = ["<CR>", "<C-M>", "<C-J>"], modes = [Mode.CMD_LINE])
|
@CommandOrMotion(keys = ["<CR>", "<C-M>", "<C-J>"], modes = [Mode.CMD_LINE])
|
||||||
public class ProcessExEntryAction : VimActionHandler.SingleExecution(), ComplicatedKeysAction {
|
public class ProcessExEntryAction : VimActionHandler.SingleExecution() {
|
||||||
override val keyStrokesSet: Set<List<KeyStroke>> =
|
|
||||||
parseKeysSet("<CR>", "<C-M>", 0x0a.toChar().toString(), 0x0d.toChar().toString())
|
|
||||||
|
|
||||||
override val type: Command.Type = Command.Type.OTHER_SELF_SYNCHRONIZED
|
override val type: Command.Type = Command.Type.OTHER_SELF_SYNCHRONIZED
|
||||||
|
|
||||||
override val flags: EnumSet<CommandFlags> = EnumSet.of(CommandFlags.FLAG_COMPLETE_EX)
|
override val flags: EnumSet<CommandFlags> = EnumSet.of(CommandFlags.FLAG_COMPLETE_EX)
|
||||||
|
@@ -27,7 +27,7 @@ public class CommandState(private val machine: VimStateMachine) {
|
|||||||
get() {
|
get() {
|
||||||
val myMode = machine.mode
|
val myMode = machine.mode
|
||||||
return when (myMode) {
|
return when (myMode) {
|
||||||
com.maddyhome.idea.vim.state.mode.Mode.CMD_LINE -> CommandState.Mode.CMD_LINE
|
is com.maddyhome.idea.vim.state.mode.Mode.CMD_LINE -> CommandState.Mode.CMD_LINE
|
||||||
com.maddyhome.idea.vim.state.mode.Mode.INSERT -> CommandState.Mode.INSERT
|
com.maddyhome.idea.vim.state.mode.Mode.INSERT -> CommandState.Mode.INSERT
|
||||||
is com.maddyhome.idea.vim.state.mode.Mode.NORMAL -> CommandState.Mode.COMMAND
|
is com.maddyhome.idea.vim.state.mode.Mode.NORMAL -> CommandState.Mode.COMMAND
|
||||||
is com.maddyhome.idea.vim.state.mode.Mode.OP_PENDING -> CommandState.Mode.OP_PENDING
|
is com.maddyhome.idea.vim.state.mode.Mode.OP_PENDING -> CommandState.Mode.OP_PENDING
|
||||||
|
@@ -9,6 +9,7 @@ package com.maddyhome.idea.vim.extension
|
|||||||
|
|
||||||
import com.intellij.openapi.application.ApplicationManager
|
import com.intellij.openapi.application.ApplicationManager
|
||||||
import com.intellij.openapi.components.service
|
import com.intellij.openapi.components.service
|
||||||
|
import com.intellij.openapi.diagnostic.logger
|
||||||
import com.intellij.openapi.editor.Editor
|
import com.intellij.openapi.editor.Editor
|
||||||
import com.maddyhome.idea.vim.KeyHandler
|
import com.maddyhome.idea.vim.KeyHandler
|
||||||
import com.maddyhome.idea.vim.VimPlugin
|
import com.maddyhome.idea.vim.VimPlugin
|
||||||
@@ -17,7 +18,6 @@ import com.maddyhome.idea.vim.api.ImmutableVimCaret
|
|||||||
import com.maddyhome.idea.vim.api.VimCaret
|
import com.maddyhome.idea.vim.api.VimCaret
|
||||||
import com.maddyhome.idea.vim.api.injector
|
import com.maddyhome.idea.vim.api.injector
|
||||||
import com.maddyhome.idea.vim.command.MappingMode
|
import com.maddyhome.idea.vim.command.MappingMode
|
||||||
import com.maddyhome.idea.vim.state.mode.SelectionType
|
|
||||||
import com.maddyhome.idea.vim.common.CommandAlias
|
import com.maddyhome.idea.vim.common.CommandAlias
|
||||||
import com.maddyhome.idea.vim.common.CommandAliasHandler
|
import com.maddyhome.idea.vim.common.CommandAliasHandler
|
||||||
import com.maddyhome.idea.vim.helper.CommandLineHelper
|
import com.maddyhome.idea.vim.helper.CommandLineHelper
|
||||||
@@ -26,6 +26,7 @@ import com.maddyhome.idea.vim.helper.vimStateMachine
|
|||||||
import com.maddyhome.idea.vim.key.MappingOwner
|
import com.maddyhome.idea.vim.key.MappingOwner
|
||||||
import com.maddyhome.idea.vim.key.OperatorFunction
|
import com.maddyhome.idea.vim.key.OperatorFunction
|
||||||
import com.maddyhome.idea.vim.newapi.vim
|
import com.maddyhome.idea.vim.newapi.vim
|
||||||
|
import com.maddyhome.idea.vim.state.mode.SelectionType
|
||||||
import com.maddyhome.idea.vim.ui.ModalEntry
|
import com.maddyhome.idea.vim.ui.ModalEntry
|
||||||
import java.awt.event.KeyEvent
|
import java.awt.event.KeyEvent
|
||||||
import javax.swing.KeyStroke
|
import javax.swing.KeyStroke
|
||||||
@@ -38,6 +39,9 @@ import javax.swing.KeyStroke
|
|||||||
* @author vlan
|
* @author vlan
|
||||||
*/
|
*/
|
||||||
public object VimExtensionFacade {
|
public object VimExtensionFacade {
|
||||||
|
|
||||||
|
private val LOG = logger<VimExtensionFacade>()
|
||||||
|
|
||||||
/** The 'map' command for mapping keys to handlers defined in extensions. */
|
/** The 'map' command for mapping keys to handlers defined in extensions. */
|
||||||
@JvmStatic
|
@JvmStatic
|
||||||
public fun putExtensionHandlerMapping(
|
public fun putExtensionHandlerMapping(
|
||||||
@@ -140,10 +144,12 @@ public object VimExtensionFacade {
|
|||||||
public fun inputKeyStroke(editor: Editor): KeyStroke {
|
public fun inputKeyStroke(editor: Editor): KeyStroke {
|
||||||
if (editor.vim.vimStateMachine.isDotRepeatInProgress) {
|
if (editor.vim.vimStateMachine.isDotRepeatInProgress) {
|
||||||
val input = Extension.consumeKeystroke()
|
val input = Extension.consumeKeystroke()
|
||||||
|
LOG.trace("inputKeyStroke: dot repeat in progress. Input: $input")
|
||||||
return input ?: error("Not enough keystrokes saved: ${Extension.lastExtensionHandler}")
|
return input ?: error("Not enough keystrokes saved: ${Extension.lastExtensionHandler}")
|
||||||
}
|
}
|
||||||
|
|
||||||
val key: KeyStroke? = if (ApplicationManager.getApplication().isUnitTestMode) {
|
val key: KeyStroke? = if (ApplicationManager.getApplication().isUnitTestMode) {
|
||||||
|
LOG.trace("Unit test mode is active")
|
||||||
val mappingStack = KeyHandler.getInstance().keyStack
|
val mappingStack = KeyHandler.getInstance().keyStack
|
||||||
mappingStack.feedSomeStroke() ?: TestInputModel.getInstance(editor).nextKeyStroke()?.also {
|
mappingStack.feedSomeStroke() ?: TestInputModel.getInstance(editor).nextKeyStroke()?.also {
|
||||||
if (editor.vim.vimStateMachine.isRecording) {
|
if (editor.vim.vimStateMachine.isRecording) {
|
||||||
@@ -151,11 +157,13 @@ public object VimExtensionFacade {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
|
LOG.trace("Getting char from the modal entry...")
|
||||||
var ref: KeyStroke? = null
|
var ref: KeyStroke? = null
|
||||||
ModalEntry.activate(editor.vim) { stroke: KeyStroke? ->
|
ModalEntry.activate(editor.vim) { stroke: KeyStroke? ->
|
||||||
ref = stroke
|
ref = stroke
|
||||||
false
|
false
|
||||||
}
|
}
|
||||||
|
LOG.trace("Got char $ref")
|
||||||
ref
|
ref
|
||||||
}
|
}
|
||||||
val result = key ?: KeyStroke.getKeyStroke(KeyEvent.VK_ESCAPE.toChar())
|
val result = key ?: KeyStroke.getKeyStroke(KeyEvent.VK_ESCAPE.toChar())
|
||||||
|
@@ -53,6 +53,11 @@ internal object VimExtensionRegistrar : VimExtensionRegistrator {
|
|||||||
@Synchronized
|
@Synchronized
|
||||||
private fun registerExtension(extensionBean: ExtensionBeanClass) {
|
private fun registerExtension(extensionBean: ExtensionBeanClass) {
|
||||||
val name = extensionBean.name ?: extensionBean.instance.name
|
val name = extensionBean.name ?: extensionBean.instance.name
|
||||||
|
if (name == "sneak" && extensionBean.name == null) {
|
||||||
|
// Filter out the old ideavim-sneak extension that used to be a separate plugin
|
||||||
|
// https://github.com/Mishkun/ideavim-sneak
|
||||||
|
return
|
||||||
|
}
|
||||||
if (name in registeredExtensions) return
|
if (name in registeredExtensions) return
|
||||||
|
|
||||||
registeredExtensions.add(name)
|
registeredExtensions.add(name)
|
||||||
|
@@ -156,11 +156,6 @@ internal class CommentaryExtension : VimExtension {
|
|||||||
private class CommentaryOperatorHandler : OperatorFunction, ExtensionHandler {
|
private class CommentaryOperatorHandler : OperatorFunction, ExtensionHandler {
|
||||||
override val isRepeatable = true
|
override val isRepeatable = true
|
||||||
|
|
||||||
// In this operator we process selection by ourselves. This is necessary for rider, VIM-1758
|
|
||||||
override fun postProcessSelection(): Boolean {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
override fun execute(editor: VimEditor, context: ExecutionContext, operatorArguments: OperatorArguments) {
|
override fun execute(editor: VimEditor, context: ExecutionContext, operatorArguments: OperatorArguments) {
|
||||||
setOperatorFunction(this)
|
setOperatorFunction(this)
|
||||||
executeNormalWithoutMapping(injector.parser.parseKeys("g@"), editor.ij)
|
executeNormalWithoutMapping(injector.parser.parseKeys("g@"), editor.ij)
|
||||||
|
@@ -246,7 +246,7 @@ internal class VimMultipleCursorsExtension : VimExtension {
|
|||||||
|
|
||||||
// Note that ignoreCase is not overridden by the `\C` in the pattern
|
// Note that ignoreCase is not overridden by the `\C` in the pattern
|
||||||
val pattern = makePattern(text, whole)
|
val pattern = makePattern(text, whole)
|
||||||
val matches = SearchHelper.findAll(editor, pattern, 0, -1, false)
|
val matches = injector.searchHelper.findAll(IjVimEditor(editor), pattern, 0, -1, false)
|
||||||
for (match in matches) {
|
for (match in matches) {
|
||||||
if (match.contains(primaryCaret.offset)) {
|
if (match.contains(primaryCaret.offset)) {
|
||||||
primaryCaret.vim.moveToOffset(match.startOffset)
|
primaryCaret.vim.moveToOffset(match.startOffset)
|
||||||
@@ -322,7 +322,7 @@ internal class VimMultipleCursorsExtension : VimExtension {
|
|||||||
searchOptions.add(SearchOptions.WRAP)
|
searchOptions.add(SearchOptions.WRAP)
|
||||||
}
|
}
|
||||||
|
|
||||||
return SearchHelper.findPattern(editor, makePattern(text, whole), startOffset, 1, searchOptions)?.startOffset ?: -1
|
return injector.searchHelper.findPattern(IjVimEditor(editor), makePattern(text, whole), startOffset, 1, searchOptions)?.startOffset ?: -1
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun makePattern(text: String, whole: Boolean): String {
|
private fun makePattern(text: String, whole: Boolean): String {
|
||||||
|
@@ -0,0 +1,291 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2003-2024 The IdeaVim authors
|
||||||
|
*
|
||||||
|
* Use of this source code is governed by an MIT-style
|
||||||
|
* license that can be found in the LICENSE.txt file or at
|
||||||
|
* https://opensource.org/licenses/MIT.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package com.maddyhome.idea.vim.extension.sneak
|
||||||
|
|
||||||
|
import com.intellij.openapi.actionSystem.DataContext
|
||||||
|
import com.intellij.openapi.application.ApplicationManager
|
||||||
|
import com.intellij.openapi.editor.Editor
|
||||||
|
import com.intellij.openapi.editor.ScrollType
|
||||||
|
import com.intellij.openapi.editor.colors.EditorColors
|
||||||
|
import com.intellij.openapi.editor.markup.EffectType
|
||||||
|
import com.intellij.openapi.editor.markup.HighlighterLayer
|
||||||
|
import com.intellij.openapi.editor.markup.HighlighterTargetArea
|
||||||
|
import com.intellij.openapi.editor.markup.RangeHighlighter
|
||||||
|
import com.intellij.openapi.editor.markup.TextAttributes
|
||||||
|
import com.intellij.openapi.util.Disposer
|
||||||
|
import com.maddyhome.idea.vim.VimProjectService
|
||||||
|
import com.maddyhome.idea.vim.api.ExecutionContext
|
||||||
|
import com.maddyhome.idea.vim.api.VimEditor
|
||||||
|
import com.maddyhome.idea.vim.api.injector
|
||||||
|
import com.maddyhome.idea.vim.api.options
|
||||||
|
import com.maddyhome.idea.vim.command.MappingMode
|
||||||
|
import com.maddyhome.idea.vim.command.OperatorArguments
|
||||||
|
import com.maddyhome.idea.vim.common.TextRange
|
||||||
|
import com.maddyhome.idea.vim.extension.ExtensionHandler
|
||||||
|
import com.maddyhome.idea.vim.extension.VimExtension
|
||||||
|
import com.maddyhome.idea.vim.extension.VimExtensionFacade
|
||||||
|
import com.maddyhome.idea.vim.extension.VimExtensionHandler
|
||||||
|
import com.maddyhome.idea.vim.helper.StrictMode
|
||||||
|
import com.maddyhome.idea.vim.newapi.ij
|
||||||
|
import java.awt.Font
|
||||||
|
import java.awt.event.KeyEvent
|
||||||
|
import java.util.concurrent.Executors
|
||||||
|
import java.util.concurrent.TimeUnit
|
||||||
|
|
||||||
|
|
||||||
|
private const val DEFAULT_HIGHLIGHT_DURATION_SNEAK: Long = 300
|
||||||
|
|
||||||
|
// By [Mikhail Levchenko](https://github.com/Mishkun)
|
||||||
|
// Original repository with the plugin: https://github.com/Mishkun/ideavim-sneak
|
||||||
|
internal class IdeaVimSneakExtension : VimExtension {
|
||||||
|
override fun getName(): String = "sneak"
|
||||||
|
|
||||||
|
override fun init() {
|
||||||
|
val highlightHandler = HighlightHandler()
|
||||||
|
mapToFunctionAndProvideKeys("s", SneakHandler(highlightHandler, Direction.FORWARD))
|
||||||
|
mapToFunctionAndProvideKeys("S", SneakHandler(highlightHandler, Direction.BACKWARD))
|
||||||
|
|
||||||
|
// workaround to support ; and , commands
|
||||||
|
mapToFunctionAndProvideKeys("f", SneakMemoryHandler("f"))
|
||||||
|
mapToFunctionAndProvideKeys("F", SneakMemoryHandler("F"))
|
||||||
|
mapToFunctionAndProvideKeys("t", SneakMemoryHandler("t"))
|
||||||
|
mapToFunctionAndProvideKeys("T", SneakMemoryHandler("T"))
|
||||||
|
|
||||||
|
mapToFunctionAndProvideKeys(";", SneakRepeatHandler(highlightHandler, RepeatDirection.IDENTICAL))
|
||||||
|
mapToFunctionAndProvideKeys(",", SneakRepeatHandler(highlightHandler, RepeatDirection.REVERSE))
|
||||||
|
}
|
||||||
|
|
||||||
|
private class SneakHandler(
|
||||||
|
private val highlightHandler: HighlightHandler,
|
||||||
|
private val direction: Direction,
|
||||||
|
) : ExtensionHandler {
|
||||||
|
override fun execute(editor: VimEditor, context: ExecutionContext, operatorArguments: OperatorArguments) {
|
||||||
|
val charone = getChar(editor) ?: return
|
||||||
|
val chartwo = getChar(editor) ?: return
|
||||||
|
val range = Util.jumpTo(editor, charone, chartwo, direction)
|
||||||
|
range?.let { highlightHandler.highlightSneakRange(editor.ij, range) }
|
||||||
|
Util.lastSymbols = "${charone}${chartwo}"
|
||||||
|
Util.lastSDirection = direction
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun getChar(editor: VimEditor): Char? {
|
||||||
|
val key = VimExtensionFacade.inputKeyStroke(editor.ij)
|
||||||
|
return when {
|
||||||
|
key.keyChar == KeyEvent.CHAR_UNDEFINED || key.keyCode == KeyEvent.VK_ESCAPE -> null
|
||||||
|
else -> key.keyChar
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This class acts as proxy for normal find commands because we need to update [Util.lastSDirection]
|
||||||
|
*/
|
||||||
|
private class SneakMemoryHandler(private val char: String) : VimExtensionHandler {
|
||||||
|
override fun execute(editor: Editor, context: DataContext) {
|
||||||
|
Util.lastSDirection = null
|
||||||
|
VimExtensionFacade.executeNormalWithoutMapping(injector.parser.parseKeys(char), editor)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private class SneakRepeatHandler(
|
||||||
|
private val highlightHandler: HighlightHandler,
|
||||||
|
private val direction: RepeatDirection,
|
||||||
|
) : ExtensionHandler {
|
||||||
|
override fun execute(editor: VimEditor, context: ExecutionContext, operatorArguments: OperatorArguments) {
|
||||||
|
val lastSDirection = Util.lastSDirection
|
||||||
|
if (lastSDirection != null) {
|
||||||
|
val (charone, chartwo) = Util.lastSymbols.toList()
|
||||||
|
val jumpRange = Util.jumpTo(editor, charone, chartwo, direction.map(lastSDirection))
|
||||||
|
jumpRange?.let { highlightHandler.highlightSneakRange(editor.ij, jumpRange) }
|
||||||
|
} else {
|
||||||
|
VimExtensionFacade.executeNormalWithoutMapping(injector.parser.parseKeys(direction.symb), editor.ij)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private object Util {
|
||||||
|
var lastSDirection: Direction? = null
|
||||||
|
var lastSymbols: String = ""
|
||||||
|
fun jumpTo(editor: VimEditor, charone: Char, chartwo: Char, sneakDirection: Direction): TextRange? {
|
||||||
|
val caret = editor.primaryCaret()
|
||||||
|
val position = caret.offset.point
|
||||||
|
val chars = editor.text()
|
||||||
|
val foundPosition = sneakDirection.findBiChar(editor, chars, position, charone, chartwo)
|
||||||
|
if (foundPosition != null) {
|
||||||
|
editor.primaryCaret().moveToOffset(foundPosition)
|
||||||
|
}
|
||||||
|
editor.ij.scrollingModel.scrollToCaret(ScrollType.MAKE_VISIBLE)
|
||||||
|
return foundPosition?.let { TextRange(foundPosition, foundPosition + 2) }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private enum class Direction(val offset: Int) {
|
||||||
|
FORWARD(1) {
|
||||||
|
override fun findBiChar(
|
||||||
|
editor: VimEditor,
|
||||||
|
charSequence: CharSequence,
|
||||||
|
position: Int,
|
||||||
|
charone: Char,
|
||||||
|
chartwo: Char
|
||||||
|
): Int? {
|
||||||
|
for (i in (position + offset) until charSequence.length - 1) {
|
||||||
|
if (matches(editor, charSequence, i, charone, chartwo)) {
|
||||||
|
return i
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return null
|
||||||
|
}
|
||||||
|
},
|
||||||
|
BACKWARD(-1) {
|
||||||
|
override fun findBiChar(
|
||||||
|
editor: VimEditor,
|
||||||
|
charSequence: CharSequence,
|
||||||
|
position: Int,
|
||||||
|
charone: Char,
|
||||||
|
chartwo: Char
|
||||||
|
): Int? {
|
||||||
|
for (i in (position + offset) downTo 0) {
|
||||||
|
if (matches(editor, charSequence, i, charone, chartwo)) {
|
||||||
|
return i
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return null
|
||||||
|
}
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
|
abstract fun findBiChar(
|
||||||
|
editor: VimEditor,
|
||||||
|
charSequence: CharSequence,
|
||||||
|
position: Int,
|
||||||
|
charone: Char,
|
||||||
|
chartwo: Char,
|
||||||
|
): Int?
|
||||||
|
|
||||||
|
fun matches(
|
||||||
|
editor: VimEditor,
|
||||||
|
charSequence: CharSequence,
|
||||||
|
charPosition: Int,
|
||||||
|
charOne: Char,
|
||||||
|
charTwo: Char,
|
||||||
|
): Boolean {
|
||||||
|
var match = charSequence[charPosition].equals(charOne, ignoreCase = injector.options(editor).ignorecase) &&
|
||||||
|
charSequence[charPosition + 1].equals(charTwo, ignoreCase = injector.options(editor).ignorecase)
|
||||||
|
|
||||||
|
if (injector.options(editor).ignorecase && injector.options(editor).smartcase) {
|
||||||
|
if (charOne.isUpperCase() || charTwo.isUpperCase()) {
|
||||||
|
match = charSequence[charPosition].equals(charOne, ignoreCase = false) &&
|
||||||
|
charSequence[charPosition + 1].equals(charTwo, ignoreCase = false)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return match
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private enum class RepeatDirection(val symb: String) {
|
||||||
|
IDENTICAL(";") {
|
||||||
|
override fun map(direction: Direction): Direction = direction
|
||||||
|
},
|
||||||
|
REVERSE(",") {
|
||||||
|
override fun map(direction: Direction): Direction = when (direction) {
|
||||||
|
Direction.FORWARD -> Direction.BACKWARD
|
||||||
|
Direction.BACKWARD -> Direction.FORWARD
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
abstract fun map(direction: Direction): Direction
|
||||||
|
}
|
||||||
|
|
||||||
|
private class HighlightHandler {
|
||||||
|
private var editor: Editor? = null
|
||||||
|
private val sneakHighlighters: MutableSet<RangeHighlighter> = mutableSetOf()
|
||||||
|
|
||||||
|
fun highlightSneakRange(editor: Editor, range: TextRange) {
|
||||||
|
clearAllSneakHighlighters()
|
||||||
|
|
||||||
|
this.editor = editor
|
||||||
|
val project = editor.project
|
||||||
|
if (project != null) {
|
||||||
|
Disposer.register(VimProjectService.getInstance(project)) {
|
||||||
|
this.editor = null
|
||||||
|
sneakHighlighters.clear()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (range.isMultiple) {
|
||||||
|
for (i in 0 until range.size()) {
|
||||||
|
highlightSingleRange(editor, range.startOffsets[i]..range.endOffsets[i])
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
highlightSingleRange(editor, range.startOffset..range.endOffset)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fun clearAllSneakHighlighters() {
|
||||||
|
sneakHighlighters.forEach { highlighter ->
|
||||||
|
editor?.markupModel?.removeHighlighter(highlighter) ?: StrictMode.fail("Highlighters without an editor")
|
||||||
|
}
|
||||||
|
|
||||||
|
sneakHighlighters.clear()
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun highlightSingleRange(editor: Editor, range: ClosedRange<Int>) {
|
||||||
|
val highlighter = editor.markupModel.addRangeHighlighter(
|
||||||
|
range.start,
|
||||||
|
range.endInclusive,
|
||||||
|
HighlighterLayer.SELECTION,
|
||||||
|
getHighlightTextAttributes(),
|
||||||
|
HighlighterTargetArea.EXACT_RANGE
|
||||||
|
)
|
||||||
|
|
||||||
|
sneakHighlighters.add(highlighter)
|
||||||
|
|
||||||
|
setClearHighlightRangeTimer(highlighter)
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun setClearHighlightRangeTimer(highlighter: RangeHighlighter) {
|
||||||
|
Executors.newSingleThreadScheduledExecutor().schedule({
|
||||||
|
ApplicationManager.getApplication().invokeLater {
|
||||||
|
editor?.markupModel?.removeHighlighter(highlighter) ?: StrictMode.fail("Highlighters without an editor")
|
||||||
|
}
|
||||||
|
}, DEFAULT_HIGHLIGHT_DURATION_SNEAK, TimeUnit.MILLISECONDS)
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun getHighlightTextAttributes() = TextAttributes(
|
||||||
|
null,
|
||||||
|
EditorColors.TEXT_SEARCH_RESULT_ATTRIBUTES.defaultAttributes.backgroundColor,
|
||||||
|
editor?.colorsScheme?.getColor(EditorColors.CARET_COLOR),
|
||||||
|
EffectType.SEARCH_MATCH,
|
||||||
|
Font.PLAIN
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Map some <Plug>(keys) command to given handler
|
||||||
|
* and create mapping to <Plug>(prefix)[keys]
|
||||||
|
*/
|
||||||
|
private fun VimExtension.mapToFunctionAndProvideKeys(keys: String, handler: ExtensionHandler) {
|
||||||
|
VimExtensionFacade.putExtensionHandlerMapping(
|
||||||
|
MappingMode.NXO,
|
||||||
|
injector.parser.parseKeys(command(keys)),
|
||||||
|
owner,
|
||||||
|
handler,
|
||||||
|
false
|
||||||
|
)
|
||||||
|
VimExtensionFacade.putKeyMapping(
|
||||||
|
MappingMode.NXO,
|
||||||
|
injector.parser.parseKeys(keys),
|
||||||
|
owner,
|
||||||
|
injector.parser.parseKeys(command(keys)),
|
||||||
|
true
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun command(keys: String) = "<Plug>(sneak-$keys)"
|
@@ -8,6 +8,7 @@
|
|||||||
package com.maddyhome.idea.vim.extension.surround
|
package com.maddyhome.idea.vim.extension.surround
|
||||||
|
|
||||||
import com.intellij.openapi.application.runWriteAction
|
import com.intellij.openapi.application.runWriteAction
|
||||||
|
import com.intellij.openapi.diagnostic.logger
|
||||||
import com.intellij.openapi.editor.Editor
|
import com.intellij.openapi.editor.Editor
|
||||||
import com.maddyhome.idea.vim.VimPlugin
|
import com.maddyhome.idea.vim.VimPlugin
|
||||||
import com.maddyhome.idea.vim.api.ExecutionContext
|
import com.maddyhome.idea.vim.api.ExecutionContext
|
||||||
@@ -18,10 +19,7 @@ import com.maddyhome.idea.vim.api.getLeadingCharacterOffset
|
|||||||
import com.maddyhome.idea.vim.api.injector
|
import com.maddyhome.idea.vim.api.injector
|
||||||
import com.maddyhome.idea.vim.api.setChangeMarks
|
import com.maddyhome.idea.vim.api.setChangeMarks
|
||||||
import com.maddyhome.idea.vim.command.MappingMode
|
import com.maddyhome.idea.vim.command.MappingMode
|
||||||
import com.maddyhome.idea.vim.state.mode.Mode
|
|
||||||
import com.maddyhome.idea.vim.command.OperatorArguments
|
import com.maddyhome.idea.vim.command.OperatorArguments
|
||||||
import com.maddyhome.idea.vim.state.mode.SelectionType
|
|
||||||
import com.maddyhome.idea.vim.state.mode.selectionType
|
|
||||||
import com.maddyhome.idea.vim.common.TextRange
|
import com.maddyhome.idea.vim.common.TextRange
|
||||||
import com.maddyhome.idea.vim.extension.ExtensionHandler
|
import com.maddyhome.idea.vim.extension.ExtensionHandler
|
||||||
import com.maddyhome.idea.vim.extension.VimExtension
|
import com.maddyhome.idea.vim.extension.VimExtension
|
||||||
@@ -33,12 +31,15 @@ import com.maddyhome.idea.vim.extension.VimExtensionFacade.putExtensionHandlerMa
|
|||||||
import com.maddyhome.idea.vim.extension.VimExtensionFacade.putKeyMappingIfMissing
|
import com.maddyhome.idea.vim.extension.VimExtensionFacade.putKeyMappingIfMissing
|
||||||
import com.maddyhome.idea.vim.extension.VimExtensionFacade.setOperatorFunction
|
import com.maddyhome.idea.vim.extension.VimExtensionFacade.setOperatorFunction
|
||||||
import com.maddyhome.idea.vim.extension.VimExtensionFacade.setRegisterForCaret
|
import com.maddyhome.idea.vim.extension.VimExtensionFacade.setRegisterForCaret
|
||||||
import com.maddyhome.idea.vim.state.mode.mode
|
|
||||||
import com.maddyhome.idea.vim.key.OperatorFunction
|
import com.maddyhome.idea.vim.key.OperatorFunction
|
||||||
import com.maddyhome.idea.vim.newapi.ij
|
import com.maddyhome.idea.vim.newapi.ij
|
||||||
import com.maddyhome.idea.vim.newapi.vim
|
import com.maddyhome.idea.vim.newapi.vim
|
||||||
import com.maddyhome.idea.vim.options.helpers.ClipboardOptionHelper
|
import com.maddyhome.idea.vim.options.helpers.ClipboardOptionHelper
|
||||||
import com.maddyhome.idea.vim.put.PutData
|
import com.maddyhome.idea.vim.put.PutData
|
||||||
|
import com.maddyhome.idea.vim.state.mode.Mode
|
||||||
|
import com.maddyhome.idea.vim.state.mode.SelectionType
|
||||||
|
import com.maddyhome.idea.vim.state.mode.mode
|
||||||
|
import com.maddyhome.idea.vim.state.mode.selectionType
|
||||||
import org.jetbrains.annotations.NonNls
|
import org.jetbrains.annotations.NonNls
|
||||||
import java.awt.event.KeyEvent
|
import java.awt.event.KeyEvent
|
||||||
import javax.swing.KeyStroke
|
import javax.swing.KeyStroke
|
||||||
@@ -249,6 +250,7 @@ internal class VimSurroundExtension : VimExtension {
|
|||||||
override fun execute(editor: VimEditor, context: ExecutionContext, operatorArguments: OperatorArguments) {
|
override fun execute(editor: VimEditor, context: ExecutionContext, operatorArguments: OperatorArguments) {
|
||||||
// Deleting surround is just changing the surrounding to "nothing"
|
// Deleting surround is just changing the surrounding to "nothing"
|
||||||
val charFrom = getChar(editor.ij)
|
val charFrom = getChar(editor.ij)
|
||||||
|
LOG.debug("DSurroundHandler: charFrom = $charFrom")
|
||||||
if (charFrom.code == 0) return
|
if (charFrom.code == 0) return
|
||||||
|
|
||||||
runWriteAction { CSurroundHandler.change(editor, context, charFrom, null) }
|
runWriteAction { CSurroundHandler.change(editor, context, charFrom, null) }
|
||||||
@@ -280,96 +282,101 @@ internal class VimSurroundExtension : VimExtension {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
companion object {
|
private val LOG = logger<VimSurroundExtension>()
|
||||||
private const val REGISTER = '"'
|
|
||||||
|
|
||||||
private val tagNameAndAttributesCapturePattern = "(\\S+)([^>]*)>".toPattern()
|
private const val REGISTER = '"'
|
||||||
|
|
||||||
private val SURROUND_PAIRS = mapOf(
|
private val tagNameAndAttributesCapturePattern = "(\\S+)([^>]*)>".toPattern()
|
||||||
'b' to ("(" to ")"),
|
|
||||||
'(' to ("( " to " )"),
|
|
||||||
')' to ("(" to ")"),
|
|
||||||
'B' to ("{" to "}"),
|
|
||||||
'{' to ("{ " to " }"),
|
|
||||||
'}' to ("{" to "}"),
|
|
||||||
'r' to ("[" to "]"),
|
|
||||||
'[' to ("[ " to " ]"),
|
|
||||||
']' to ("[" to "]"),
|
|
||||||
'a' to ("<" to ">"),
|
|
||||||
'>' to ("<" to ">"),
|
|
||||||
's' to (" " to ""),
|
|
||||||
)
|
|
||||||
|
|
||||||
private fun getSurroundPair(c: Char): Pair<String, String>? = if (c in SURROUND_PAIRS) {
|
private val SURROUND_PAIRS = mapOf(
|
||||||
SURROUND_PAIRS[c]
|
'b' to ("(" to ")"),
|
||||||
} else if (!c.isLetter()) {
|
'(' to ("( " to " )"),
|
||||||
val s = c.toString()
|
')' to ("(" to ")"),
|
||||||
s to s
|
'B' to ("{" to "}"),
|
||||||
} else {
|
'{' to ("{ " to " }"),
|
||||||
null
|
'}' to ("{" to "}"),
|
||||||
}
|
'r' to ("[" to "]"),
|
||||||
|
'[' to ("[ " to " ]"),
|
||||||
|
']' to ("[" to "]"),
|
||||||
|
'a' to ("<" to ">"),
|
||||||
|
'>' to ("<" to ">"),
|
||||||
|
's' to (" " to ""),
|
||||||
|
)
|
||||||
|
|
||||||
private fun inputTagPair(editor: Editor): Pair<String, String>? {
|
private fun getSurroundPair(c: Char): Pair<String, String>? = if (c in SURROUND_PAIRS) {
|
||||||
val tagInput = inputString(editor, "<", '>')
|
SURROUND_PAIRS[c]
|
||||||
val matcher = tagNameAndAttributesCapturePattern.matcher(tagInput)
|
} else if (!c.isLetter()) {
|
||||||
return if (matcher.find()) {
|
val s = c.toString()
|
||||||
val tagName = matcher.group(1)
|
s to s
|
||||||
val tagAttributes = matcher.group(2)
|
} else {
|
||||||
"<$tagName$tagAttributes>" to "</$tagName>"
|
null
|
||||||
} else {
|
}
|
||||||
null
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private fun inputFunctionName(
|
private fun inputTagPair(editor: Editor): Pair<String, String>? {
|
||||||
editor: Editor,
|
val tagInput = inputString(editor, "<", '>')
|
||||||
withInternalSpaces: Boolean,
|
val matcher = tagNameAndAttributesCapturePattern.matcher(tagInput)
|
||||||
): Pair<String, String>? {
|
return if (matcher.find()) {
|
||||||
val functionNameInput = inputString(editor, "function: ", null)
|
val tagName = matcher.group(1)
|
||||||
if (functionNameInput.isEmpty()) return null
|
val tagAttributes = matcher.group(2)
|
||||||
return if (withInternalSpaces) "$functionNameInput( " to " )" else "$functionNameInput(" to ")"
|
"<$tagName$tagAttributes>" to "</$tagName>"
|
||||||
}
|
} else {
|
||||||
|
null
|
||||||
private fun getOrInputPair(c: Char, editor: Editor): Pair<String, String>? = when (c) {
|
}
|
||||||
'<', 't' -> inputTagPair(editor)
|
}
|
||||||
'f' -> inputFunctionName(editor, false)
|
|
||||||
'F' -> inputFunctionName(editor, true)
|
private fun inputFunctionName(
|
||||||
else -> getSurroundPair(c)
|
editor: Editor,
|
||||||
}
|
withInternalSpaces: Boolean,
|
||||||
|
): Pair<String, String>? {
|
||||||
private fun getChar(editor: Editor): Char {
|
val functionNameInput = inputString(editor, "function: ", null)
|
||||||
val key = inputKeyStroke(editor)
|
if (functionNameInput.isEmpty()) return null
|
||||||
val keyChar = key.keyChar
|
return if (withInternalSpaces) "$functionNameInput( " to " )" else "$functionNameInput(" to ")"
|
||||||
return if (keyChar == KeyEvent.CHAR_UNDEFINED || keyChar.code == KeyEvent.VK_ESCAPE) {
|
}
|
||||||
0.toChar()
|
|
||||||
} else {
|
private fun getOrInputPair(c: Char, editor: Editor): Pair<String, String>? = when (c) {
|
||||||
keyChar
|
'<', 't' -> inputTagPair(editor)
|
||||||
}
|
'f' -> inputFunctionName(editor, false)
|
||||||
}
|
'F' -> inputFunctionName(editor, true)
|
||||||
|
else -> getSurroundPair(c)
|
||||||
private fun performSurround(pair: Pair<String, String>, range: TextRange, caret: VimCaret, tagsOnNewLines: Boolean = false) {
|
}
|
||||||
runWriteAction {
|
|
||||||
val editor = caret.editor
|
private fun getChar(editor: Editor): Char {
|
||||||
val change = VimPlugin.getChange()
|
val key = inputKeyStroke(editor)
|
||||||
val leftSurround = pair.first + if (tagsOnNewLines) "\n" else ""
|
val keyChar = key.keyChar
|
||||||
|
val res = if (keyChar == KeyEvent.CHAR_UNDEFINED || keyChar.code == KeyEvent.VK_ESCAPE) {
|
||||||
val isEOF = range.endOffset == editor.text().length
|
0.toChar()
|
||||||
val hasNewLine = editor.endsWithNewLine()
|
} else {
|
||||||
val rightSurround = if (tagsOnNewLines) {
|
keyChar
|
||||||
if (isEOF && !hasNewLine) {
|
}
|
||||||
"\n" + pair.second
|
LOG.trace("getChar: $res")
|
||||||
} else {
|
return res
|
||||||
pair.second + "\n"
|
}
|
||||||
}
|
|
||||||
} else {
|
private fun performSurround(pair: Pair<String, String>, range: TextRange, caret: VimCaret, tagsOnNewLines: Boolean = false) {
|
||||||
pair.second
|
runWriteAction {
|
||||||
}
|
val editor = caret.editor
|
||||||
|
val change = VimPlugin.getChange()
|
||||||
change.insertText(editor, caret, range.startOffset, leftSurround)
|
val leftSurround = pair.first + if (tagsOnNewLines) "\n" else ""
|
||||||
change.insertText(editor, caret, range.endOffset + leftSurround.length, rightSurround)
|
|
||||||
injector.markService.setChangeMarks(caret, TextRange(range.startOffset, range.endOffset + leftSurround.length + rightSurround.length))
|
val isEOF = range.endOffset == editor.text().length
|
||||||
}
|
val hasNewLine = editor.endsWithNewLine()
|
||||||
}
|
val rightSurround = if (tagsOnNewLines) {
|
||||||
|
if (isEOF && !hasNewLine) {
|
||||||
|
"\n" + pair.second
|
||||||
|
} else {
|
||||||
|
pair.second + "\n"
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
pair.second
|
||||||
|
}
|
||||||
|
|
||||||
|
change.insertText(editor, caret, range.startOffset, leftSurround)
|
||||||
|
change.insertText(editor, caret, range.endOffset + leftSurround.length, rightSurround)
|
||||||
|
injector.markService.setChangeMarks(
|
||||||
|
caret,
|
||||||
|
TextRange(range.startOffset, range.endOffset + leftSurround.length + rightSurround.length)
|
||||||
|
)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -20,9 +20,6 @@ import com.intellij.openapi.editor.actions.EnterAction
|
|||||||
import com.intellij.openapi.editor.event.EditorMouseEvent
|
import com.intellij.openapi.editor.event.EditorMouseEvent
|
||||||
import com.intellij.openapi.editor.event.EditorMouseListener
|
import com.intellij.openapi.editor.event.EditorMouseListener
|
||||||
import com.intellij.openapi.editor.impl.TextRangeInterval
|
import com.intellij.openapi.editor.impl.TextRangeInterval
|
||||||
import com.intellij.openapi.ui.MessageType
|
|
||||||
import com.intellij.openapi.ui.popup.Balloon
|
|
||||||
import com.intellij.openapi.ui.popup.JBPopupFactory
|
|
||||||
import com.intellij.openapi.util.UserDataHolder
|
import com.intellij.openapi.util.UserDataHolder
|
||||||
import com.intellij.openapi.util.text.StringUtil
|
import com.intellij.openapi.util.text.StringUtil
|
||||||
import com.intellij.psi.codeStyle.CodeStyleManager
|
import com.intellij.psi.codeStyle.CodeStyleManager
|
||||||
@@ -65,7 +62,6 @@ import com.maddyhome.idea.vim.helper.endOffsetInclusive
|
|||||||
import com.maddyhome.idea.vim.helper.inInsertMode
|
import com.maddyhome.idea.vim.helper.inInsertMode
|
||||||
import com.maddyhome.idea.vim.helper.moveToInlayAwareLogicalPosition
|
import com.maddyhome.idea.vim.helper.moveToInlayAwareLogicalPosition
|
||||||
import com.maddyhome.idea.vim.helper.moveToInlayAwareOffset
|
import com.maddyhome.idea.vim.helper.moveToInlayAwareOffset
|
||||||
import com.maddyhome.idea.vim.icons.VimIcons
|
|
||||||
import com.maddyhome.idea.vim.key.KeyHandlerKeeper.Companion.getInstance
|
import com.maddyhome.idea.vim.key.KeyHandlerKeeper.Companion.getInstance
|
||||||
import com.maddyhome.idea.vim.listener.VimInsertListener
|
import com.maddyhome.idea.vim.listener.VimInsertListener
|
||||||
import com.maddyhome.idea.vim.newapi.IjEditorExecutionContext
|
import com.maddyhome.idea.vim.newapi.IjEditorExecutionContext
|
||||||
@@ -89,7 +85,6 @@ import kotlin.math.min
|
|||||||
*/
|
*/
|
||||||
public class ChangeGroup : VimChangeGroupBase() {
|
public class ChangeGroup : VimChangeGroupBase() {
|
||||||
private val insertListeners = ContainerUtil.createLockFreeCopyOnWriteList<VimInsertListener>()
|
private val insertListeners = ContainerUtil.createLockFreeCopyOnWriteList<VimInsertListener>()
|
||||||
private var lastShownTime = 0L
|
|
||||||
private val listener: EditorMouseListener = object : EditorMouseListener {
|
private val listener: EditorMouseListener = object : EditorMouseListener {
|
||||||
override fun mouseClicked(event: EditorMouseEvent) {
|
override fun mouseClicked(event: EditorMouseEvent) {
|
||||||
val editor = event.editor
|
val editor = event.editor
|
||||||
@@ -103,10 +98,6 @@ public class ChangeGroup : VimChangeGroupBase() {
|
|||||||
EventFacade.getInstance().addEditorMouseListener(editor!!, listener, disposable)
|
EventFacade.getInstance().addEditorMouseListener(editor!!, listener, disposable)
|
||||||
}
|
}
|
||||||
|
|
||||||
public fun editorReleased(editor: Editor?) {
|
|
||||||
EventFacade.getInstance().removeEditorMouseListener(editor!!, listener)
|
|
||||||
}
|
|
||||||
|
|
||||||
override fun type(vimEditor: VimEditor, context: ExecutionContext, key: Char) {
|
override fun type(vimEditor: VimEditor, context: ExecutionContext, key: Char) {
|
||||||
val editor = (vimEditor as IjVimEditor).editor
|
val editor = (vimEditor as IjVimEditor).editor
|
||||||
val ijContext = context.ij
|
val ijContext = context.ij
|
||||||
@@ -645,25 +636,6 @@ public class ChangeGroup : VimChangeGroupBase() {
|
|||||||
avalanche: Boolean,
|
avalanche: Boolean,
|
||||||
): Boolean {
|
): Boolean {
|
||||||
|
|
||||||
// Just an easter egg
|
|
||||||
if (avalanche) {
|
|
||||||
val currentTime = System.currentTimeMillis()
|
|
||||||
if (currentTime - lastShownTime > 60000) {
|
|
||||||
lastShownTime = currentTime
|
|
||||||
ApplicationManager.getApplication().invokeLater {
|
|
||||||
val balloon = JBPopupFactory.getInstance()
|
|
||||||
.createHtmlTextBalloonBuilder(
|
|
||||||
"Wow, nice vim skills!", VimIcons.IDEAVIM,
|
|
||||||
MessageType.INFO.titleForeground, MessageType.INFO.popupBackground,
|
|
||||||
null
|
|
||||||
).createBalloon()
|
|
||||||
balloon.show(
|
|
||||||
JBPopupFactory.getInstance().guessBestPopupLocation((editor as IjVimEditor).editor),
|
|
||||||
Balloon.Position.below
|
|
||||||
)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
val nf: List<String> = injector.options(editor).nrformats
|
val nf: List<String> = injector.options(editor).nrformats
|
||||||
val alpha = nf.contains("alpha")
|
val alpha = nf.contains("alpha")
|
||||||
val hex = nf.contains("hex")
|
val hex = nf.contains("hex")
|
||||||
|
@@ -29,6 +29,7 @@ public open class GlobalIjOptions(scope: OptionAccessScope) : OptionsPropertiesB
|
|||||||
public val lookupkeys: StringListOptionValue by optionProperty(IjOptions.lookupkeys)
|
public val lookupkeys: StringListOptionValue by optionProperty(IjOptions.lookupkeys)
|
||||||
public var trackactionids: Boolean by optionProperty(IjOptions.trackactionids)
|
public var trackactionids: Boolean by optionProperty(IjOptions.trackactionids)
|
||||||
public var visualdelay: Int by optionProperty(IjOptions.visualdelay)
|
public var visualdelay: Int by optionProperty(IjOptions.visualdelay)
|
||||||
|
public var showmodewidget: Boolean by optionProperty(IjOptions.showmodewidget)
|
||||||
|
|
||||||
// Temporary options to control work-in-progress behaviour
|
// Temporary options to control work-in-progress behaviour
|
||||||
public var oldundo: Boolean by optionProperty(IjOptions.oldundo)
|
public var oldundo: Boolean by optionProperty(IjOptions.oldundo)
|
||||||
@@ -36,6 +37,7 @@ public open class GlobalIjOptions(scope: OptionAccessScope) : OptionsPropertiesB
|
|||||||
public var exCommandAnnotation: Boolean by optionProperty(IjOptions.exCommandAnnotation)
|
public var exCommandAnnotation: Boolean by optionProperty(IjOptions.exCommandAnnotation)
|
||||||
public var vimscriptFunctionAnnotation: Boolean by optionProperty(IjOptions.vimscriptFunctionAnnotation)
|
public var vimscriptFunctionAnnotation: Boolean by optionProperty(IjOptions.vimscriptFunctionAnnotation)
|
||||||
public var commandOrMotionAnnotation: Boolean by optionProperty(IjOptions.commandOrMotionAnnotation)
|
public var commandOrMotionAnnotation: Boolean by optionProperty(IjOptions.commandOrMotionAnnotation)
|
||||||
|
public var useNewRegex: Boolean by optionProperty(IjOptions.useNewRegex)
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@@ -86,6 +86,8 @@ public object IjOptions {
|
|||||||
public val oldundo: ToggleOption = addOption(ToggleOption("oldundo", GLOBAL, "oldundo", false, isTemporary = true))
|
public val oldundo: ToggleOption = addOption(ToggleOption("oldundo", GLOBAL, "oldundo", false, isTemporary = true))
|
||||||
public val vimscriptFunctionAnnotation: ToggleOption = addOption(ToggleOption("vimscriptfunctionannotation", GLOBAL, "vimscriptfunctionannotation", true, isTemporary = true))
|
public val vimscriptFunctionAnnotation: ToggleOption = addOption(ToggleOption("vimscriptfunctionannotation", GLOBAL, "vimscriptfunctionannotation", true, isTemporary = true))
|
||||||
public val commandOrMotionAnnotation: ToggleOption = addOption(ToggleOption("commandormotionannotation", GLOBAL, "commandormotionannotation", true, isTemporary = true))
|
public val commandOrMotionAnnotation: ToggleOption = addOption(ToggleOption("commandormotionannotation", GLOBAL, "commandormotionannotation", true, isTemporary = true))
|
||||||
|
public val showmodewidget: ToggleOption = addOption(ToggleOption("showmodewidget", GLOBAL, "showmodewidget", false, isTemporary = true))
|
||||||
|
public val useNewRegex: ToggleOption = addOption(ToggleOption("usenewregex", GLOBAL, "usenewregex", true, isTemporary = true))
|
||||||
|
|
||||||
// This needs to be Option<out VimDataType> so that it can work with derived option types, such as NumberOption, which
|
// This needs to be Option<out VimDataType> so that it can work with derived option types, such as NumberOption, which
|
||||||
// derives from Option<VimInt>
|
// derives from Option<VimInt>
|
||||||
|
@@ -24,17 +24,14 @@ import com.intellij.openapi.keymap.KeymapManager;
|
|||||||
import com.intellij.openapi.keymap.ex.KeymapManagerEx;
|
import com.intellij.openapi.keymap.ex.KeymapManagerEx;
|
||||||
import com.maddyhome.idea.vim.EventFacade;
|
import com.maddyhome.idea.vim.EventFacade;
|
||||||
import com.maddyhome.idea.vim.VimPlugin;
|
import com.maddyhome.idea.vim.VimPlugin;
|
||||||
import com.maddyhome.idea.vim.action.ComplicatedKeysAction;
|
|
||||||
import com.maddyhome.idea.vim.action.VimShortcutKeyAction;
|
import com.maddyhome.idea.vim.action.VimShortcutKeyAction;
|
||||||
import com.maddyhome.idea.vim.action.change.LazyVimCommand;
|
import com.maddyhome.idea.vim.action.change.LazyVimCommand;
|
||||||
import com.maddyhome.idea.vim.api.*;
|
import com.maddyhome.idea.vim.api.*;
|
||||||
import com.maddyhome.idea.vim.command.MappingMode;
|
import com.maddyhome.idea.vim.command.MappingMode;
|
||||||
import com.maddyhome.idea.vim.ex.ExOutputModel;
|
import com.maddyhome.idea.vim.ex.ExOutputModel;
|
||||||
import com.maddyhome.idea.vim.handler.EditorActionHandlerBase;
|
|
||||||
import com.maddyhome.idea.vim.helper.HelperKt;
|
import com.maddyhome.idea.vim.helper.HelperKt;
|
||||||
import com.maddyhome.idea.vim.key.*;
|
import com.maddyhome.idea.vim.key.*;
|
||||||
import com.maddyhome.idea.vim.newapi.IjNativeAction;
|
import com.maddyhome.idea.vim.newapi.IjNativeAction;
|
||||||
import com.maddyhome.idea.vim.newapi.IjVimActionsInitiator;
|
|
||||||
import com.maddyhome.idea.vim.newapi.IjVimEditor;
|
import com.maddyhome.idea.vim.newapi.IjVimEditor;
|
||||||
import kotlin.Pair;
|
import kotlin.Pair;
|
||||||
import kotlin.text.StringsKt;
|
import kotlin.text.StringsKt;
|
||||||
@@ -49,6 +46,7 @@ import java.awt.event.KeyEvent;
|
|||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.*;
|
import java.util.*;
|
||||||
|
|
||||||
|
import static com.maddyhome.idea.vim.api.VimInjectorKt.injector;
|
||||||
import static java.util.stream.Collectors.toList;
|
import static java.util.stream.Collectors.toList;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -221,63 +219,20 @@ public class KeyGroup extends VimKeyGroupBase implements PersistentStateComponen
|
|||||||
registerRequiredShortcut(keyStrokes, MappingOwner.IdeaVim.System.INSTANCE);
|
registerRequiredShortcut(keyStrokes, MappingOwner.IdeaVim.System.INSTANCE);
|
||||||
|
|
||||||
for (MappingMode mappingMode : command.getModes()) {
|
for (MappingMode mappingMode : command.getModes()) {
|
||||||
Node<VimActionsInitiator> node = getKeyRoot(mappingMode);
|
Node<LazyVimCommand> node = getKeyRoot(mappingMode);
|
||||||
NodesKt.addLeafs(node, keyStrokes, command);
|
NodesKt.addLeafs(node, keyStrokes, command);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@Deprecated
|
|
||||||
public void registerCommandAction(@NotNull VimActionsInitiator actionHolder) {
|
|
||||||
IjVimActionsInitiator holder = (IjVimActionsInitiator)actionHolder;
|
|
||||||
|
|
||||||
if (!VimPlugin.getPluginId().equals(holder.getBean().getPluginDescriptor().getPluginId())) {
|
|
||||||
logger.error("IdeaVim doesn't accept contributions to `vimActions` extension points. " +
|
|
||||||
"Please create a plugin using `VimExtension`. " +
|
|
||||||
"Plugin to blame: " +
|
|
||||||
holder.getBean().getPluginDescriptor().getPluginId());
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
Set<List<KeyStroke>> actionKeys = holder.getBean().getParsedKeys();
|
|
||||||
if (actionKeys == null) {
|
|
||||||
final EditorActionHandlerBase action = actionHolder.getInstance();
|
|
||||||
if (action instanceof ComplicatedKeysAction) {
|
|
||||||
actionKeys = ((ComplicatedKeysAction)action).getKeyStrokesSet();
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
throw new RuntimeException("Cannot register action: " + action.getClass().getName());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Set<MappingMode> actionModes = holder.getBean().getParsedModes();
|
|
||||||
if (actionModes == null) {
|
|
||||||
throw new RuntimeException("Cannot register action: " + holder.getBean().getImplementation());
|
|
||||||
}
|
|
||||||
|
|
||||||
if (ApplicationManager.getApplication().isUnitTestMode()) {
|
|
||||||
initIdentityChecker();
|
|
||||||
for (List<KeyStroke> keys : actionKeys) {
|
|
||||||
checkCommand(actionModes, actionHolder.getInstance(), keys);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
for (List<KeyStroke> keyStrokes : actionKeys) {
|
|
||||||
registerRequiredShortcut(keyStrokes, MappingOwner.IdeaVim.System.INSTANCE);
|
|
||||||
|
|
||||||
for (MappingMode mappingMode : actionModes) {
|
|
||||||
Node<VimActionsInitiator> node = getKeyRoot(mappingMode);
|
|
||||||
NodesKt.addLeafs(node, keyStrokes, actionHolder);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private void registerRequiredShortcut(@NotNull List<KeyStroke> keys, MappingOwner owner) {
|
private void registerRequiredShortcut(@NotNull List<KeyStroke> keys, MappingOwner owner) {
|
||||||
for (KeyStroke key : keys) {
|
for (KeyStroke key : keys) {
|
||||||
if (key.getKeyChar() == KeyEvent.CHAR_UNDEFINED &&
|
if (key.getKeyChar() == KeyEvent.CHAR_UNDEFINED) {
|
||||||
key.getKeyCode() != KeyEvent.VK_ESCAPE &&
|
if (!injector.getOptionGroup().getGlobalOptions().getOctopushandler() ||
|
||||||
key.getKeyCode() != KeyEvent.VK_ENTER) {
|
!(key.getKeyCode() == KeyEvent.VK_ESCAPE && key.getModifiers() == 0) &&
|
||||||
getRequiredShortcutKeys().add(new RequiredShortcut(key, owner));
|
!(key.getKeyCode() == KeyEvent.VK_ENTER && key.getModifiers() == 0)) {
|
||||||
|
getRequiredShortcutKeys().add(new RequiredShortcut(key, owner));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -7,6 +7,8 @@
|
|||||||
*/
|
*/
|
||||||
package com.maddyhome.idea.vim.group
|
package com.maddyhome.idea.vim.group
|
||||||
|
|
||||||
|
import com.intellij.codeInsight.completion.CompletionPhase
|
||||||
|
import com.intellij.codeInsight.completion.impl.CompletionServiceImpl
|
||||||
import com.intellij.openapi.application.ApplicationManager
|
import com.intellij.openapi.application.ApplicationManager
|
||||||
import com.intellij.openapi.diagnostic.logger
|
import com.intellij.openapi.diagnostic.logger
|
||||||
import com.intellij.openapi.progress.ProcessCanceledException
|
import com.intellij.openapi.progress.ProcessCanceledException
|
||||||
@@ -61,22 +63,33 @@ internal class MacroGroup : VimMacroBase() {
|
|||||||
try {
|
try {
|
||||||
myPotemkinProgress.text2 = if (isInternalMacro) "Executing internal macro" else ""
|
myPotemkinProgress.text2 = if (isInternalMacro) "Executing internal macro" else ""
|
||||||
val runnable = runnable@{
|
val runnable = runnable@{
|
||||||
// Handle one keystroke then queue up the next key
|
try {
|
||||||
for (i in 0 until total) {
|
// Handle one keystroke then queue up the next key
|
||||||
myPotemkinProgress.fraction = (i + 1).toDouble() / total
|
for (i in 0 until total) {
|
||||||
while (keyStack.hasStroke()) {
|
|
||||||
val key = keyStack.feedStroke()
|
|
||||||
try {
|
try {
|
||||||
myPotemkinProgress.checkCanceled()
|
myPotemkinProgress.fraction = (i + 1).toDouble() / total
|
||||||
} catch (e: ProcessCanceledException) {
|
while (keyStack.hasStroke()) {
|
||||||
return@runnable
|
val key = keyStack.feedStroke()
|
||||||
|
try {
|
||||||
|
myPotemkinProgress.checkCanceled()
|
||||||
|
} catch (e: ProcessCanceledException) {
|
||||||
|
return@runnable
|
||||||
|
}
|
||||||
|
ProgressManager.getInstance().executeNonCancelableSection {
|
||||||
|
// Prevent autocompletion during macros.
|
||||||
|
// See https://github.com/JetBrains/ideavim/pull/772 for details
|
||||||
|
CompletionServiceImpl.setCompletionPhase(CompletionPhase.NoCompletion)
|
||||||
|
getInstance().handleKey(editor, key, context)
|
||||||
|
}
|
||||||
|
if (injector.messages.isError()) return@runnable
|
||||||
|
}
|
||||||
|
} finally {
|
||||||
|
keyStack.resetFirst()
|
||||||
}
|
}
|
||||||
ProgressManager.getInstance().executeNonCancelableSection { getInstance().handleKey(editor, key, context) }
|
|
||||||
if (injector.messages.isError()) return@runnable
|
|
||||||
}
|
}
|
||||||
keyStack.resetFirst()
|
} finally {
|
||||||
|
keyStack.removeFirst()
|
||||||
}
|
}
|
||||||
keyStack.removeFirst()
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (isInternalMacro) {
|
if (isInternalMacro) {
|
||||||
|
@@ -1,138 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright 2003-2023 The IdeaVim authors
|
|
||||||
*
|
|
||||||
* Use of this source code is governed by an MIT-style
|
|
||||||
* license that can be found in the LICENSE.txt file or at
|
|
||||||
* https://opensource.org/licenses/MIT.
|
|
||||||
*/
|
|
||||||
|
|
||||||
package com.maddyhome.idea.vim.group;
|
|
||||||
|
|
||||||
import com.intellij.ide.bookmark.LineBookmark;
|
|
||||||
import com.intellij.openapi.editor.Editor;
|
|
||||||
import com.maddyhome.idea.vim.api.*;
|
|
||||||
import com.maddyhome.idea.vim.common.TextRange;
|
|
||||||
import com.maddyhome.idea.vim.mark.IntellijMark;
|
|
||||||
import com.maddyhome.idea.vim.mark.Jump;
|
|
||||||
import com.maddyhome.idea.vim.mark.Mark;
|
|
||||||
import com.maddyhome.idea.vim.newapi.IjVimEditor;
|
|
||||||
import org.jetbrains.annotations.ApiStatus;
|
|
||||||
import org.jetbrains.annotations.NotNull;
|
|
||||||
import org.jetbrains.annotations.Nullable;
|
|
||||||
|
|
||||||
import java.util.ArrayList;
|
|
||||||
import java.util.HashMap;
|
|
||||||
import java.util.List;
|
|
||||||
import java.util.Set;
|
|
||||||
|
|
||||||
@Deprecated
|
|
||||||
@ApiStatus.ScheduledForRemoval(inVersion = "2.3")
|
|
||||||
public class MarkGroup {
|
|
||||||
public List<Jump> jumps = VimInjectorKt.injector.getJumpService().getJumps("");
|
|
||||||
|
|
||||||
public void saveJumpLocation(@NotNull Editor editor) {
|
|
||||||
VimInjectorKt.injector.getJumpService().saveJumpLocation(new IjVimEditor(editor));
|
|
||||||
}
|
|
||||||
|
|
||||||
public void saveJumpLocation(@NotNull VimEditor editor) {
|
|
||||||
VimInjectorKt.injector.getJumpService().saveJumpLocation(editor);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void setChangeMarks(@NotNull VimEditor vimEditor, @NotNull TextRange range) {
|
|
||||||
VimMarkService markService = VimInjectorKt.injector.getMarkService();
|
|
||||||
VimMarkServiceKt.setChangeMarks(markService, vimEditor.primaryCaret(), range);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void addJump(@NotNull VimEditor editor, boolean reset) {
|
|
||||||
VimJumpServiceKt.addJump(VimInjectorKt.injector.getJumpService(), editor, reset);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Nullable
|
|
||||||
public Mark getMark(@NotNull VimEditor editor, char ch) {
|
|
||||||
return VimInjectorKt.injector.getMarkService().getMark(editor.primaryCaret(), ch);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Nullable
|
|
||||||
public Jump getJump(int count) {
|
|
||||||
return VimInjectorKt.injector.getJumpService().getJump("", count);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Nullable
|
|
||||||
public Mark createSystemMark(char ch, int line, int col, @NotNull VimEditor editor) {
|
|
||||||
Editor ijEditor = ((IjVimEditor)editor).getEditor();
|
|
||||||
@Nullable LineBookmark systemMark = SystemMarks.createOrGetSystemMark(ch, line, ijEditor);
|
|
||||||
if (systemMark == null) {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
return new IntellijMark(systemMark, col, ijEditor.getProject());
|
|
||||||
}
|
|
||||||
|
|
||||||
public boolean setMark(@NotNull VimEditor editor, char ch, int offset) {
|
|
||||||
return VimInjectorKt.injector.getMarkService().setMark(editor.primaryCaret(), ch, offset);
|
|
||||||
}
|
|
||||||
|
|
||||||
public boolean setMark(@NotNull VimEditor editor, char ch) {
|
|
||||||
return VimInjectorKt.injector.getMarkService().setMark(editor, ch);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void includeCurrentCommandAsNavigation(@NotNull VimEditor editor) {
|
|
||||||
VimInjectorKt.injector.getJumpService().includeCurrentCommandAsNavigation(editor);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Nullable
|
|
||||||
public Mark getFileMark(@NotNull VimEditor editor, char ch) {
|
|
||||||
return VimInjectorKt.injector.getMarkService().getMark(editor.primaryCaret(), ch);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void setVisualSelectionMarks(@NotNull VimEditor editor, @NotNull TextRange range) {
|
|
||||||
VimMarkService markService = VimInjectorKt.injector.getMarkService();
|
|
||||||
VimMarkServiceKt.setVisualSelectionMarks(markService, editor.primaryCaret(), range);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Nullable
|
|
||||||
public TextRange getChangeMarks(@NotNull VimEditor editor) {
|
|
||||||
return VimInjectorKt.injector.getMarkService().getChangeMarks(editor.primaryCaret());
|
|
||||||
}
|
|
||||||
|
|
||||||
@Nullable
|
|
||||||
public TextRange getVisualSelectionMarks(@NotNull VimEditor editor) {
|
|
||||||
return VimInjectorKt.injector.getMarkService().getVisualSelectionMarks(editor.primaryCaret());
|
|
||||||
}
|
|
||||||
|
|
||||||
public void resetAllMarks() {
|
|
||||||
VimInjectorKt.injector.getMarkService().resetAllMarks();
|
|
||||||
}
|
|
||||||
|
|
||||||
public void removeMark(char ch, @NotNull Mark mark) {
|
|
||||||
VimInjectorKt.injector.getMarkService().removeMark(ch, mark);
|
|
||||||
}
|
|
||||||
|
|
||||||
@NotNull
|
|
||||||
public List<Mark> getMarks(@NotNull VimEditor editor) {
|
|
||||||
Set<Mark> marks = VimInjectorKt.injector.getMarkService().getAllLocalMarks(editor.primaryCaret());
|
|
||||||
marks.addAll(VimInjectorKt.injector.getMarkService().getGlobalMarks(editor));
|
|
||||||
return new ArrayList<>(marks);
|
|
||||||
}
|
|
||||||
|
|
||||||
public int getJumpSpot() {
|
|
||||||
return VimInjectorKt.injector.getJumpService().getJumpSpot("");
|
|
||||||
}
|
|
||||||
|
|
||||||
public void updateMarkFromDelete(@Nullable VimEditor editor,
|
|
||||||
@Nullable HashMap<Character, Mark> marks,
|
|
||||||
int delStartOff,
|
|
||||||
int delLength) {
|
|
||||||
VimInjectorKt.injector.getMarkService().updateMarksFromDelete(editor, delStartOff, delLength);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void updateMarkFromInsert(@Nullable VimEditor editor,
|
|
||||||
@Nullable HashMap<Character, Mark> marks,
|
|
||||||
int insStartOff,
|
|
||||||
int insLength) {
|
|
||||||
VimInjectorKt.injector.getMarkService().updateMarksFromInsert(editor, insStartOff, insLength);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void dropLastJump() {
|
|
||||||
VimInjectorKt.injector.getJumpService().dropLastJump("");
|
|
||||||
}
|
|
||||||
}
|
|
@@ -48,9 +48,7 @@ import com.maddyhome.idea.vim.api.options
|
|||||||
import com.maddyhome.idea.vim.api.visualLineToBufferLine
|
import com.maddyhome.idea.vim.api.visualLineToBufferLine
|
||||||
import com.maddyhome.idea.vim.command.Argument
|
import com.maddyhome.idea.vim.command.Argument
|
||||||
import com.maddyhome.idea.vim.command.MotionType
|
import com.maddyhome.idea.vim.command.MotionType
|
||||||
import com.maddyhome.idea.vim.state.mode.Mode
|
|
||||||
import com.maddyhome.idea.vim.command.OperatorArguments
|
import com.maddyhome.idea.vim.command.OperatorArguments
|
||||||
import com.maddyhome.idea.vim.state.VimStateMachine
|
|
||||||
import com.maddyhome.idea.vim.common.TextRange
|
import com.maddyhome.idea.vim.common.TextRange
|
||||||
import com.maddyhome.idea.vim.ex.ExOutputModel
|
import com.maddyhome.idea.vim.ex.ExOutputModel
|
||||||
import com.maddyhome.idea.vim.handler.Motion
|
import com.maddyhome.idea.vim.handler.Motion
|
||||||
@@ -74,6 +72,8 @@ import com.maddyhome.idea.vim.newapi.IjVimCaret
|
|||||||
import com.maddyhome.idea.vim.newapi.IjVimEditor
|
import com.maddyhome.idea.vim.newapi.IjVimEditor
|
||||||
import com.maddyhome.idea.vim.newapi.ij
|
import com.maddyhome.idea.vim.newapi.ij
|
||||||
import com.maddyhome.idea.vim.newapi.vim
|
import com.maddyhome.idea.vim.newapi.vim
|
||||||
|
import com.maddyhome.idea.vim.state.VimStateMachine
|
||||||
|
import com.maddyhome.idea.vim.state.mode.Mode
|
||||||
import com.maddyhome.idea.vim.ui.ex.ExEntryPanel
|
import com.maddyhome.idea.vim.ui.ex.ExEntryPanel
|
||||||
import org.jetbrains.annotations.Range
|
import org.jetbrains.annotations.Range
|
||||||
import java.io.File
|
import java.io.File
|
||||||
@@ -461,11 +461,13 @@ internal class MotionGroup : VimMotionGroupBase() {
|
|||||||
val fileEditor = event.oldEditor
|
val fileEditor = event.oldEditor
|
||||||
if (fileEditor is TextEditor) {
|
if (fileEditor is TextEditor) {
|
||||||
val editor = fileEditor.editor
|
val editor = fileEditor.editor
|
||||||
ExOutputModel.getInstance(editor).clear()
|
if (!editor.isDisposed) {
|
||||||
editor.vim.let { vimEditor ->
|
ExOutputModel.getInstance(editor).clear()
|
||||||
if (VimStateMachine.getInstance(vimEditor).mode is Mode.VISUAL) {
|
editor.vim.let { vimEditor ->
|
||||||
vimEditor.exitVisualMode()
|
if (VimStateMachine.getInstance(vimEditor).mode is Mode.VISUAL) {
|
||||||
KeyHandler.getInstance().reset(vimEditor)
|
vimEditor.exitVisualMode()
|
||||||
|
KeyHandler.getInstance().reset(vimEditor)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -21,8 +21,11 @@ import com.intellij.openapi.actionSystem.ActionUpdateThread
|
|||||||
import com.intellij.openapi.actionSystem.AnAction
|
import com.intellij.openapi.actionSystem.AnAction
|
||||||
import com.intellij.openapi.actionSystem.AnActionEvent
|
import com.intellij.openapi.actionSystem.AnActionEvent
|
||||||
import com.intellij.openapi.actionSystem.KeyboardShortcut
|
import com.intellij.openapi.actionSystem.KeyboardShortcut
|
||||||
|
import com.intellij.openapi.diagnostic.logger
|
||||||
import com.intellij.openapi.ide.CopyPasteManager
|
import com.intellij.openapi.ide.CopyPasteManager
|
||||||
import com.intellij.openapi.keymap.KeymapUtil
|
import com.intellij.openapi.keymap.KeymapUtil
|
||||||
|
import com.intellij.openapi.keymap.ex.KeymapManagerEx
|
||||||
|
import com.intellij.openapi.keymap.impl.ui.KeymapPanel
|
||||||
import com.intellij.openapi.options.ShowSettingsUtil
|
import com.intellij.openapi.options.ShowSettingsUtil
|
||||||
import com.intellij.openapi.project.DumbAwareAction
|
import com.intellij.openapi.project.DumbAwareAction
|
||||||
import com.intellij.openapi.project.Project
|
import com.intellij.openapi.project.Project
|
||||||
@@ -32,6 +35,7 @@ import com.maddyhome.idea.vim.VimPlugin
|
|||||||
import com.maddyhome.idea.vim.api.VimEditor
|
import com.maddyhome.idea.vim.api.VimEditor
|
||||||
import com.maddyhome.idea.vim.api.globalOptions
|
import com.maddyhome.idea.vim.api.globalOptions
|
||||||
import com.maddyhome.idea.vim.api.injector
|
import com.maddyhome.idea.vim.api.injector
|
||||||
|
import com.maddyhome.idea.vim.handler.KeyMapIssue
|
||||||
import com.maddyhome.idea.vim.helper.MessageHelper
|
import com.maddyhome.idea.vim.helper.MessageHelper
|
||||||
import com.maddyhome.idea.vim.key.ShortcutOwner
|
import com.maddyhome.idea.vim.key.ShortcutOwner
|
||||||
import com.maddyhome.idea.vim.key.ShortcutOwnerInfo
|
import com.maddyhome.idea.vim.key.ShortcutOwnerInfo
|
||||||
@@ -180,6 +184,77 @@ internal class NotificationService(private val project: Project?) {
|
|||||||
ActionIdNotifier.notifyActionId(id, project)
|
ActionIdNotifier.notifyActionId(id, project)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fun notifyKeymapIssues(issues: ArrayList<KeyMapIssue>) {
|
||||||
|
val keymapManager = KeymapManagerEx.getInstanceEx()
|
||||||
|
val keymap = keymapManager.activeKeymap
|
||||||
|
val message = buildString {
|
||||||
|
appendLine("Current IDE keymap (${keymap.name}) has issues:<br/>")
|
||||||
|
issues.forEach {
|
||||||
|
when (it) {
|
||||||
|
is KeyMapIssue.AddShortcut -> {
|
||||||
|
appendLine("- ${it.key} key is not assigned to the ${it.action} action.<br/>")
|
||||||
|
}
|
||||||
|
is KeyMapIssue.RemoveShortcut -> {
|
||||||
|
appendLine("- ${it.shortcut} key is incorrectly assigned to the ${it.action} action.<br/>")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
val notification = IDEAVIM_STICKY_GROUP.createNotification(
|
||||||
|
IDEAVIM_NOTIFICATION_TITLE,
|
||||||
|
message,
|
||||||
|
NotificationType.ERROR,
|
||||||
|
)
|
||||||
|
notification.subtitle = "IDE keymap misconfigured"
|
||||||
|
notification.addAction(object : DumbAwareAction("Fix Keymap") {
|
||||||
|
override fun actionPerformed(e: AnActionEvent) {
|
||||||
|
issues.forEach {
|
||||||
|
when (it) {
|
||||||
|
is KeyMapIssue.AddShortcut -> {
|
||||||
|
keymap.addShortcut(it.actionId, KeyboardShortcut(it.keyStroke, null))
|
||||||
|
}
|
||||||
|
|
||||||
|
is KeyMapIssue.RemoveShortcut -> {
|
||||||
|
keymap.removeShortcut(it.actionId, it.shortcut)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
LOG.info("Shortcuts updated $issues")
|
||||||
|
notification.expire()
|
||||||
|
requiredShortcutsAssigned()
|
||||||
|
}
|
||||||
|
})
|
||||||
|
notification.addAction(object : DumbAwareAction("Open Keymap Settings") {
|
||||||
|
override fun actionPerformed(e: AnActionEvent) {
|
||||||
|
ShowSettingsUtil.getInstance().showSettingsDialog(e.project, KeymapPanel::class.java)
|
||||||
|
notification.hideBalloon()
|
||||||
|
}
|
||||||
|
})
|
||||||
|
notification.addAction(object : DumbAwareAction("Ignore") {
|
||||||
|
override fun actionPerformed(e: AnActionEvent) {
|
||||||
|
LOG.info("Ignored to update shortcuts $issues")
|
||||||
|
notification.hideBalloon()
|
||||||
|
}
|
||||||
|
})
|
||||||
|
notification.notify(project)
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun requiredShortcutsAssigned() {
|
||||||
|
val notification = Notification(
|
||||||
|
IDEAVIM_NOTIFICATION_ID,
|
||||||
|
IDEAVIM_NOTIFICATION_TITLE,
|
||||||
|
"Keymap fixed",
|
||||||
|
NotificationType.INFORMATION,
|
||||||
|
)
|
||||||
|
notification.addAction(object : DumbAwareAction("Open Keymap Settings") {
|
||||||
|
override fun actionPerformed(e: AnActionEvent) {
|
||||||
|
ShowSettingsUtil.getInstance().showSettingsDialog(e.project, KeymapPanel::class.java)
|
||||||
|
notification.hideBalloon()
|
||||||
|
}
|
||||||
|
})
|
||||||
|
notification.notify(project)
|
||||||
|
}
|
||||||
|
|
||||||
object ActionIdNotifier {
|
object ActionIdNotifier {
|
||||||
private var notification: Notification? = null
|
private var notification: Notification? = null
|
||||||
private const val NO_ID = "<i>Cannot detect action id</i>"
|
private const val NO_ID = "<i>Cannot detect action id</i>"
|
||||||
@@ -314,6 +389,8 @@ internal class NotificationService(private val project: Project?) {
|
|||||||
const val IDEAVIM_NOTIFICATION_TITLE = "IdeaVim"
|
const val IDEAVIM_NOTIFICATION_TITLE = "IdeaVim"
|
||||||
const val ideajoinExamplesUrl = "https://jb.gg/f9zji9"
|
const val ideajoinExamplesUrl = "https://jb.gg/f9zji9"
|
||||||
|
|
||||||
|
private val LOG = logger<NotificationService>()
|
||||||
|
|
||||||
private fun createIdeaVimRcManually(message: String, project: Project?) {
|
private fun createIdeaVimRcManually(message: String, project: Project?) {
|
||||||
val notification =
|
val notification =
|
||||||
Notification(IDEAVIM_NOTIFICATION_ID, IDEAVIM_NOTIFICATION_TITLE, message, NotificationType.WARNING)
|
Notification(IDEAVIM_NOTIFICATION_ID, IDEAVIM_NOTIFICATION_TITLE, message, NotificationType.WARNING)
|
||||||
|
@@ -40,10 +40,6 @@ internal class OptionGroup : VimOptionGroupBase(), IjVimOptionGroup {
|
|||||||
override fun getGlobalIjOptions() = GlobalIjOptions(OptionAccessScope.GLOBAL(null))
|
override fun getGlobalIjOptions() = GlobalIjOptions(OptionAccessScope.GLOBAL(null))
|
||||||
override fun getEffectiveIjOptions(editor: VimEditor) = EffectiveIjOptions(OptionAccessScope.EFFECTIVE(editor))
|
override fun getEffectiveIjOptions(editor: VimEditor) = EffectiveIjOptions(OptionAccessScope.EFFECTIVE(editor))
|
||||||
|
|
||||||
private fun updateFallbackWindow(fallbackWindow: VimEditor, targetEditor: VimEditor) {
|
|
||||||
copyPerWindowGlobalValues(fallbackWindow, targetEditor)
|
|
||||||
}
|
|
||||||
|
|
||||||
companion object {
|
companion object {
|
||||||
fun fileEditorManagerSelectionChangedCallback(event: FileEditorManagerEvent) {
|
fun fileEditorManagerSelectionChangedCallback(event: FileEditorManagerEvent) {
|
||||||
// Vim only has one window, and it's not possible to close it. This means that editing a new file will always
|
// Vim only has one window, and it's not possible to close it. This means that editing a new file will always
|
||||||
@@ -58,6 +54,8 @@ internal class OptionGroup : VimOptionGroupBase(), IjVimOptionGroup {
|
|||||||
// Unfortunately, we can't reliably know if a closing editor is the selected editor. Instead, we rely on selection
|
// Unfortunately, we can't reliably know if a closing editor is the selected editor. Instead, we rely on selection
|
||||||
// change events. If an editor is losing selection and there is no new selection, we can assume this means that
|
// change events. If an editor is losing selection and there is no new selection, we can assume this means that
|
||||||
// the last editor has been closed, and use the closed editor to update the fallback window
|
// the last editor has been closed, and use the closed editor to update the fallback window
|
||||||
|
//
|
||||||
|
// XXX: event.oldEditor will must probably return a disposed editor. So, it should be treated with care
|
||||||
if (event.newEditor == null) {
|
if (event.newEditor == null) {
|
||||||
(event.oldEditor as? TextEditor)?.editor?.let {
|
(event.oldEditor as? TextEditor)?.editor?.let {
|
||||||
(VimPlugin.getOptionGroup() as OptionGroup).updateFallbackWindow(injector.fallbackWindow, it.vim)
|
(VimPlugin.getOptionGroup() as OptionGroup).updateFallbackWindow(injector.fallbackWindow, it.vim)
|
||||||
@@ -67,28 +65,28 @@ internal class OptionGroup : VimOptionGroupBase(), IjVimOptionGroup {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
internal class IjOptionConstants {
|
public class IjOptionConstants {
|
||||||
@Suppress("SpellCheckingInspection", "MemberVisibilityCanBePrivate")
|
@Suppress("SpellCheckingInspection", "MemberVisibilityCanBePrivate", "ConstPropertyName")
|
||||||
companion object {
|
public companion object {
|
||||||
|
|
||||||
const val idearefactormode_keep = "keep"
|
public const val idearefactormode_keep: String = "keep"
|
||||||
const val idearefactormode_select = "select"
|
public const val idearefactormode_select: String = "select"
|
||||||
const val idearefactormode_visual = "visual"
|
public const val idearefactormode_visual: String = "visual"
|
||||||
|
|
||||||
const val ideastatusicon_enabled = "enabled"
|
public const val ideastatusicon_enabled: String = "enabled"
|
||||||
const val ideastatusicon_gray = "gray"
|
public const val ideastatusicon_gray: String = "gray"
|
||||||
const val ideastatusicon_disabled = "disabled"
|
public const val ideastatusicon_disabled: String = "disabled"
|
||||||
|
|
||||||
const val ideavimsupport_dialog = "dialog"
|
public const val ideavimsupport_dialog: String = "dialog"
|
||||||
const val ideavimsupport_singleline = "singleline"
|
public const val ideavimsupport_singleline: String = "singleline"
|
||||||
const val ideavimsupport_dialoglegacy = "dialoglegacy"
|
public const val ideavimsupport_dialoglegacy: String = "dialoglegacy"
|
||||||
|
|
||||||
const val ideawrite_all = "all"
|
public const val ideawrite_all: String = "all"
|
||||||
const val ideawrite_file = "file"
|
public const val ideawrite_file: String = "file"
|
||||||
|
|
||||||
val ideaStatusIconValues = setOf(ideastatusicon_enabled, ideastatusicon_gray, ideastatusicon_disabled)
|
public val ideaStatusIconValues: Set<String> = setOf(ideastatusicon_enabled, ideastatusicon_gray, ideastatusicon_disabled)
|
||||||
val ideaRefactorModeValues = setOf(idearefactormode_keep, idearefactormode_select, idearefactormode_visual)
|
public val ideaRefactorModeValues: Set<String> = setOf(idearefactormode_keep, idearefactormode_select, idearefactormode_visual)
|
||||||
val ideaWriteValues = setOf(ideawrite_all, ideawrite_file)
|
public val ideaWriteValues: Set<String> = setOf(ideawrite_all, ideawrite_file)
|
||||||
val ideavimsupportValues = setOf(ideavimsupport_dialog, ideavimsupport_singleline, ideavimsupport_dialoglegacy)
|
public val ideavimsupportValues: Set<String> = setOf(ideavimsupport_dialog, ideavimsupport_singleline, ideavimsupport_dialoglegacy)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -1,285 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright 2003-2023 The IdeaVim authors
|
|
||||||
*
|
|
||||||
* Use of this source code is governed by an MIT-style
|
|
||||||
* license that can be found in the LICENSE.txt file or at
|
|
||||||
* https://opensource.org/licenses/MIT.
|
|
||||||
*/
|
|
||||||
|
|
||||||
package com.maddyhome.idea.vim.group;
|
|
||||||
|
|
||||||
import com.intellij.execution.ExecutionException;
|
|
||||||
import com.intellij.execution.configurations.GeneralCommandLine;
|
|
||||||
import com.intellij.execution.process.CapturingProcessHandler;
|
|
||||||
import com.intellij.execution.process.ProcessAdapter;
|
|
||||||
import com.intellij.execution.process.ProcessEvent;
|
|
||||||
import com.intellij.execution.process.ProcessOutput;
|
|
||||||
import com.intellij.openapi.actionSystem.DataContext;
|
|
||||||
import com.intellij.openapi.diagnostic.Logger;
|
|
||||||
import com.intellij.openapi.editor.Editor;
|
|
||||||
import com.intellij.openapi.progress.ProcessCanceledException;
|
|
||||||
import com.intellij.openapi.progress.ProgressIndicator;
|
|
||||||
import com.intellij.openapi.progress.ProgressIndicatorProvider;
|
|
||||||
import com.intellij.openapi.progress.ProgressManager;
|
|
||||||
import com.intellij.util.execution.ParametersListUtil;
|
|
||||||
import com.intellij.util.text.CharSequenceReader;
|
|
||||||
import com.maddyhome.idea.vim.KeyHandler;
|
|
||||||
import com.maddyhome.idea.vim.VimPlugin;
|
|
||||||
import com.maddyhome.idea.vim.api.ExecutionContext;
|
|
||||||
import com.maddyhome.idea.vim.api.VimEditor;
|
|
||||||
import com.maddyhome.idea.vim.api.VimInjectorKt;
|
|
||||||
import com.maddyhome.idea.vim.api.VimProcessGroupBase;
|
|
||||||
import com.maddyhome.idea.vim.command.Command;
|
|
||||||
import com.maddyhome.idea.vim.state.mode.Mode;
|
|
||||||
import com.maddyhome.idea.vim.state.VimStateMachine;
|
|
||||||
import com.maddyhome.idea.vim.ex.ExException;
|
|
||||||
import com.maddyhome.idea.vim.ex.InvalidCommandException;
|
|
||||||
import com.maddyhome.idea.vim.helper.UiHelper;
|
|
||||||
import com.maddyhome.idea.vim.newapi.IjEditorExecutionContext;
|
|
||||||
import com.maddyhome.idea.vim.newapi.IjVimEditor;
|
|
||||||
import com.maddyhome.idea.vim.ui.ex.ExEntryPanel;
|
|
||||||
import com.maddyhome.idea.vim.vimscript.model.CommandLineVimLContext;
|
|
||||||
import org.jetbrains.annotations.NotNull;
|
|
||||||
import org.jetbrains.annotations.Nullable;
|
|
||||||
|
|
||||||
import javax.swing.*;
|
|
||||||
import java.io.*;
|
|
||||||
import java.util.ArrayList;
|
|
||||||
|
|
||||||
import static com.maddyhome.idea.vim.api.VimInjectorKt.globalOptions;
|
|
||||||
import static com.maddyhome.idea.vim.api.VimInjectorKt.injector;
|
|
||||||
|
|
||||||
|
|
||||||
public class ProcessGroup extends VimProcessGroupBase {
|
|
||||||
public String getLastCommand() {
|
|
||||||
return lastCommand;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void startSearchCommand(@NotNull VimEditor editor, ExecutionContext context, int count, char leader) {
|
|
||||||
if (((IjVimEditor)editor).getEditor().isOneLineMode()) // Don't allow searching in one line editors
|
|
||||||
{
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
String initText = "";
|
|
||||||
String label = String.valueOf(leader);
|
|
||||||
|
|
||||||
ExEntryPanel panel = ExEntryPanel.getInstance();
|
|
||||||
panel.activate(((IjVimEditor)editor).getEditor(), ((DataContext)context.getContext()), label, initText, count);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public @NotNull String endSearchCommand() {
|
|
||||||
ExEntryPanel panel = ExEntryPanel.getInstance();
|
|
||||||
panel.deactivate(true);
|
|
||||||
|
|
||||||
return panel.getText();
|
|
||||||
}
|
|
||||||
|
|
||||||
public void startExCommand(@NotNull VimEditor editor, ExecutionContext context, @NotNull Command cmd) {
|
|
||||||
// Don't allow ex commands in one line editors
|
|
||||||
if (editor.isOneLineMode()) return;
|
|
||||||
|
|
||||||
String initText = getRange(((IjVimEditor) editor).getEditor(), cmd);
|
|
||||||
injector.getMarkService().setVisualSelectionMarks(editor);
|
|
||||||
VimStateMachine.Companion.getInstance(editor).setMode(Mode.CMD_LINE.INSTANCE);
|
|
||||||
ExEntryPanel panel = ExEntryPanel.getInstance();
|
|
||||||
panel.activate(((IjVimEditor) editor).getEditor(), ((IjEditorExecutionContext) context).getContext(), ":", initText, 1);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public boolean processExKey(@NotNull VimEditor editor, @NotNull KeyStroke stroke) {
|
|
||||||
// This will only get called if somehow the key focus ended up in the editor while the ex entry window
|
|
||||||
// is open. So I'll put focus back in the editor and process the key.
|
|
||||||
|
|
||||||
ExEntryPanel panel = ExEntryPanel.getInstance();
|
|
||||||
if (panel.isActive()) {
|
|
||||||
UiHelper.requestFocus(panel.getEntry());
|
|
||||||
panel.handleKey(stroke);
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
VimStateMachine.Companion.getInstance(editor).setMode(new Mode.NORMAL());
|
|
||||||
KeyHandler.getInstance().reset(editor);
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public boolean processExEntry(final @NotNull VimEditor editor, final @NotNull ExecutionContext context) {
|
|
||||||
ExEntryPanel panel = ExEntryPanel.getInstance();
|
|
||||||
panel.deactivate(true);
|
|
||||||
boolean res = true;
|
|
||||||
try {
|
|
||||||
VimStateMachine.Companion.getInstance(editor).setMode(new Mode.NORMAL());
|
|
||||||
|
|
||||||
logger.debug("processing command");
|
|
||||||
|
|
||||||
final String text = panel.getText();
|
|
||||||
|
|
||||||
if (!panel.getLabel().equals(":")) {
|
|
||||||
// Search is handled via Argument.Type.EX_STRING. Although ProcessExEntryAction is registered as the handler for
|
|
||||||
// <CR> in both command and search modes, it's only invoked for command mode (see KeyHandler.handleCommandNode).
|
|
||||||
// We should never be invoked for anything other than an actual ex command.
|
|
||||||
throw new InvalidCommandException("Expected ':' command. Got '" + panel.getLabel() + "'", text);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (logger.isDebugEnabled()) logger.debug("swing=" + SwingUtilities.isEventDispatchThread());
|
|
||||||
|
|
||||||
VimInjectorKt.getInjector().getVimscriptExecutor().execute(text, editor, context, skipHistory(editor), true, CommandLineVimLContext.INSTANCE);
|
|
||||||
}
|
|
||||||
catch (ExException e) {
|
|
||||||
VimPlugin.showMessage(e.getMessage());
|
|
||||||
VimPlugin.indicateError();
|
|
||||||
res = false;
|
|
||||||
}
|
|
||||||
catch (Exception bad) {
|
|
||||||
ProcessGroup.logger.error(bad);
|
|
||||||
VimPlugin.indicateError();
|
|
||||||
res = false;
|
|
||||||
}
|
|
||||||
|
|
||||||
return res;
|
|
||||||
}
|
|
||||||
|
|
||||||
// commands executed from map command / macro should not be added to history
|
|
||||||
private boolean skipHistory(VimEditor editor) {
|
|
||||||
return VimStateMachine.Companion.getInstance(editor).getMappingState().isExecutingMap() || injector.getMacro().isExecutingMacro();
|
|
||||||
}
|
|
||||||
|
|
||||||
public void cancelExEntry(final @NotNull VimEditor editor, boolean resetCaret) {
|
|
||||||
VimStateMachine.Companion.getInstance(editor).setMode(new Mode.NORMAL());
|
|
||||||
KeyHandler.getInstance().reset(editor);
|
|
||||||
ExEntryPanel panel = ExEntryPanel.getInstance();
|
|
||||||
panel.deactivate(true, resetCaret);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void startFilterCommand(@NotNull VimEditor editor, ExecutionContext context, @NotNull Command cmd) {
|
|
||||||
String initText = getRange(((IjVimEditor) editor).getEditor(), cmd) + "!";
|
|
||||||
VimStateMachine.Companion.getInstance(editor).setMode(Mode.CMD_LINE.INSTANCE);
|
|
||||||
ExEntryPanel panel = ExEntryPanel.getInstance();
|
|
||||||
panel.activate(((IjVimEditor) editor).getEditor(), ((IjEditorExecutionContext) context).getContext(), ":", initText, 1);
|
|
||||||
}
|
|
||||||
|
|
||||||
private @NotNull String getRange(Editor editor, @NotNull Command cmd) {
|
|
||||||
String initText = "";
|
|
||||||
if (VimStateMachine.Companion.getInstance(new IjVimEditor(editor)).getMode() instanceof Mode.VISUAL) {
|
|
||||||
initText = "'<,'>";
|
|
||||||
}
|
|
||||||
else if (cmd.getRawCount() > 0) {
|
|
||||||
if (cmd.getCount() == 1) {
|
|
||||||
initText = ".";
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
initText = ".,.+" + (cmd.getCount() - 1);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return initText;
|
|
||||||
}
|
|
||||||
|
|
||||||
public @Nullable String executeCommand(@NotNull VimEditor editor, @NotNull String command, @Nullable CharSequence input, @Nullable String currentDirectoryPath)
|
|
||||||
throws ExecutionException, ProcessCanceledException {
|
|
||||||
|
|
||||||
// This is a much simplified version of how Vim does this. We're using stdin/stdout directly, while Vim will
|
|
||||||
// redirect to temp files ('shellredir' and 'shelltemp') or use pipes. We don't support 'shellquote', because we're
|
|
||||||
// not handling redirection, but we do use 'shellxquote' and 'shellxescape', because these have defaults that work
|
|
||||||
// better with Windows. We also don't bother using ShellExecute for Windows commands beginning with `start`.
|
|
||||||
// Finally, we're also not bothering with the crazy space and backslash handling of the 'shell' options content.
|
|
||||||
return ProgressManager.getInstance().runProcessWithProgressSynchronously(() -> {
|
|
||||||
|
|
||||||
final String shell = globalOptions(injector).getShell();
|
|
||||||
final String shellcmdflag = globalOptions(injector).getShellcmdflag();
|
|
||||||
final String shellxescape = globalOptions(injector).getShellxescape();
|
|
||||||
final String shellxquote = globalOptions(injector).getShellxquote();
|
|
||||||
|
|
||||||
// For Win32. See :help 'shellxescape'
|
|
||||||
final String escapedCommand = shellxquote.equals("(")
|
|
||||||
? doEscape(command, shellxescape, "^")
|
|
||||||
: command;
|
|
||||||
// Required for Win32+cmd.exe, defaults to "(". See :help 'shellxquote'
|
|
||||||
final String quotedCommand = shellxquote.equals("(")
|
|
||||||
? "(" + escapedCommand + ")"
|
|
||||||
: (shellxquote.equals("\"(")
|
|
||||||
? "\"(" + escapedCommand + ")\""
|
|
||||||
: shellxquote + escapedCommand + shellxquote);
|
|
||||||
|
|
||||||
final ArrayList<String> commands = new ArrayList<>();
|
|
||||||
commands.add(shell);
|
|
||||||
if (!shellcmdflag.isEmpty()) {
|
|
||||||
// Note that Vim also does a simple whitespace split for multiple parameters
|
|
||||||
commands.addAll(ParametersListUtil.parse(shellcmdflag));
|
|
||||||
}
|
|
||||||
commands.add(quotedCommand);
|
|
||||||
|
|
||||||
if (logger.isDebugEnabled()) {
|
|
||||||
logger.debug(String.format("shell=%s shellcmdflag=%s command=%s", shell, shellcmdflag, quotedCommand));
|
|
||||||
}
|
|
||||||
|
|
||||||
final GeneralCommandLine commandLine = new GeneralCommandLine(commands);
|
|
||||||
if (currentDirectoryPath != null) {
|
|
||||||
commandLine.setWorkDirectory(currentDirectoryPath);
|
|
||||||
}
|
|
||||||
final CapturingProcessHandler handler = new CapturingProcessHandler(commandLine);
|
|
||||||
if (input != null) {
|
|
||||||
handler.addProcessListener(new ProcessAdapter() {
|
|
||||||
@Override
|
|
||||||
public void startNotified(@NotNull ProcessEvent event) {
|
|
||||||
try {
|
|
||||||
final CharSequenceReader charSequenceReader = new CharSequenceReader(input);
|
|
||||||
final BufferedWriter outputStreamWriter = new BufferedWriter(new OutputStreamWriter(handler.getProcessInput()));
|
|
||||||
copy(charSequenceReader, outputStreamWriter);
|
|
||||||
outputStreamWriter.close();
|
|
||||||
}
|
|
||||||
catch (IOException e) {
|
|
||||||
logger.error(e);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
final ProgressIndicator progressIndicator = ProgressIndicatorProvider.getInstance().getProgressIndicator();
|
|
||||||
final ProcessOutput output = handler.runProcessWithProgressIndicator(progressIndicator);
|
|
||||||
|
|
||||||
lastCommand = command;
|
|
||||||
|
|
||||||
if (output.isCancelled()) {
|
|
||||||
// TODO: Vim will use whatever text has already been written to stdout
|
|
||||||
// For whatever reason, we're not getting any here, so just throw an exception
|
|
||||||
throw new ProcessCanceledException();
|
|
||||||
}
|
|
||||||
|
|
||||||
final Integer exitCode = handler.getExitCode();
|
|
||||||
if (exitCode != null && exitCode != 0) {
|
|
||||||
VimPlugin.showMessage("shell returned " + exitCode);
|
|
||||||
VimPlugin.indicateError();
|
|
||||||
}
|
|
||||||
|
|
||||||
// Get stderr; stdout and strip colors, which are not handles properly.
|
|
||||||
return (output.getStderr() + output.getStdout()).replaceAll("\u001B\\[[;\\d]*m", "");
|
|
||||||
}, "IdeaVim - !" + command, true, ((IjVimEditor) editor).getEditor().getProject());
|
|
||||||
}
|
|
||||||
|
|
||||||
private String doEscape(String original, String charsToEscape, String escapeChar) {
|
|
||||||
String result = original;
|
|
||||||
for (char c : charsToEscape.toCharArray()) {
|
|
||||||
result = result.replace("" + c, escapeChar + c);
|
|
||||||
}
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
// TODO: Java 10 has a transferTo method we could use instead
|
|
||||||
private void copy(@NotNull Reader from, @NotNull Writer to) throws IOException {
|
|
||||||
char[] buf = new char[2048];
|
|
||||||
int cnt;
|
|
||||||
while ((cnt = from.read(buf)) != -1) {
|
|
||||||
to.write(buf, 0, cnt);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private String lastCommand;
|
|
||||||
|
|
||||||
private static final Logger logger = Logger.getInstance(ProcessGroup.class.getName());
|
|
||||||
}
|
|
288
src/main/java/com/maddyhome/idea/vim/group/ProcessGroup.kt
Normal file
288
src/main/java/com/maddyhome/idea/vim/group/ProcessGroup.kt
Normal file
@@ -0,0 +1,288 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2003-2023 The IdeaVim authors
|
||||||
|
*
|
||||||
|
* Use of this source code is governed by an MIT-style
|
||||||
|
* license that can be found in the LICENSE.txt file or at
|
||||||
|
* https://opensource.org/licenses/MIT.
|
||||||
|
*/
|
||||||
|
package com.maddyhome.idea.vim.group
|
||||||
|
|
||||||
|
import com.intellij.execution.ExecutionException
|
||||||
|
import com.intellij.execution.configurations.GeneralCommandLine
|
||||||
|
import com.intellij.execution.process.CapturingProcessHandler
|
||||||
|
import com.intellij.execution.process.ProcessAdapter
|
||||||
|
import com.intellij.execution.process.ProcessEvent
|
||||||
|
import com.intellij.openapi.diagnostic.debug
|
||||||
|
import com.intellij.openapi.diagnostic.logger
|
||||||
|
import com.intellij.openapi.progress.ProcessCanceledException
|
||||||
|
import com.intellij.openapi.progress.ProgressIndicatorProvider
|
||||||
|
import com.intellij.openapi.progress.ProgressManager
|
||||||
|
import com.intellij.util.execution.ParametersListUtil
|
||||||
|
import com.intellij.util.text.CharSequenceReader
|
||||||
|
import com.maddyhome.idea.vim.KeyHandler.Companion.getInstance
|
||||||
|
import com.maddyhome.idea.vim.VimPlugin
|
||||||
|
import com.maddyhome.idea.vim.api.ExecutionContext
|
||||||
|
import com.maddyhome.idea.vim.api.VimEditor
|
||||||
|
import com.maddyhome.idea.vim.api.VimProcessGroupBase
|
||||||
|
import com.maddyhome.idea.vim.api.globalOptions
|
||||||
|
import com.maddyhome.idea.vim.api.injector
|
||||||
|
import com.maddyhome.idea.vim.command.Command
|
||||||
|
import com.maddyhome.idea.vim.ex.ExException
|
||||||
|
import com.maddyhome.idea.vim.ex.InvalidCommandException
|
||||||
|
import com.maddyhome.idea.vim.helper.requestFocus
|
||||||
|
import com.maddyhome.idea.vim.helper.vimStateMachine
|
||||||
|
import com.maddyhome.idea.vim.newapi.ij
|
||||||
|
import com.maddyhome.idea.vim.state.VimStateMachine.Companion.getInstance
|
||||||
|
import com.maddyhome.idea.vim.state.mode.Mode
|
||||||
|
import com.maddyhome.idea.vim.state.mode.Mode.NORMAL
|
||||||
|
import com.maddyhome.idea.vim.state.mode.Mode.VISUAL
|
||||||
|
import com.maddyhome.idea.vim.state.mode.ReturnableFromCmd
|
||||||
|
import com.maddyhome.idea.vim.state.mode.mode
|
||||||
|
import com.maddyhome.idea.vim.ui.ex.ExEntryPanel
|
||||||
|
import com.maddyhome.idea.vim.vimscript.model.CommandLineVimLContext
|
||||||
|
import java.io.BufferedWriter
|
||||||
|
import java.io.IOException
|
||||||
|
import java.io.OutputStreamWriter
|
||||||
|
import java.io.Reader
|
||||||
|
import java.io.Writer
|
||||||
|
import javax.swing.KeyStroke
|
||||||
|
import javax.swing.SwingUtilities
|
||||||
|
|
||||||
|
public class ProcessGroup : VimProcessGroupBase() {
|
||||||
|
override var lastCommand: String? = null
|
||||||
|
private set
|
||||||
|
override var isCommandProcessing: Boolean = false
|
||||||
|
override var modeBeforeCommandProcessing: Mode? = null
|
||||||
|
|
||||||
|
public override fun startSearchCommand(editor: VimEditor, context: ExecutionContext, count: Int, leader: Char) {
|
||||||
|
// Don't allow searching in one line editors
|
||||||
|
if (editor.isOneLineMode()) return
|
||||||
|
|
||||||
|
val initText = ""
|
||||||
|
val label = leader.toString()
|
||||||
|
|
||||||
|
val panel = ExEntryPanel.getInstance()
|
||||||
|
panel.activate(editor.ij, context.ij, label, initText, count)
|
||||||
|
}
|
||||||
|
|
||||||
|
public override fun endSearchCommand(): String {
|
||||||
|
val panel = ExEntryPanel.getInstance()
|
||||||
|
panel.deactivate(true)
|
||||||
|
|
||||||
|
return panel.text
|
||||||
|
}
|
||||||
|
|
||||||
|
public override fun startExCommand(editor: VimEditor, context: ExecutionContext, cmd: Command) {
|
||||||
|
// Don't allow ex commands in one line editors
|
||||||
|
if (editor.isOneLineMode()) return
|
||||||
|
|
||||||
|
val currentMode = editor.vimStateMachine.mode
|
||||||
|
check(currentMode is ReturnableFromCmd) {
|
||||||
|
"Cannot enable cmd mode from current mode $currentMode"
|
||||||
|
}
|
||||||
|
|
||||||
|
isCommandProcessing = true
|
||||||
|
modeBeforeCommandProcessing = currentMode
|
||||||
|
val initText = getRange(editor, cmd)
|
||||||
|
injector.markService.setVisualSelectionMarks(editor)
|
||||||
|
editor.vimStateMachine.mode = Mode.CMD_LINE(currentMode)
|
||||||
|
val panel = ExEntryPanel.getInstance()
|
||||||
|
panel.activate(editor.ij, context.ij, ":", initText, 1)
|
||||||
|
}
|
||||||
|
|
||||||
|
public override fun processExKey(editor: VimEditor, stroke: KeyStroke): Boolean {
|
||||||
|
// This will only get called if somehow the key focus ended up in the editor while the ex entry window
|
||||||
|
// is open. So I'll put focus back in the editor and process the key.
|
||||||
|
|
||||||
|
val panel = ExEntryPanel.getInstance()
|
||||||
|
if (panel.isActive) {
|
||||||
|
requestFocus(panel.entry)
|
||||||
|
panel.handleKey(stroke)
|
||||||
|
|
||||||
|
return true
|
||||||
|
} else {
|
||||||
|
getInstance(editor).mode = NORMAL()
|
||||||
|
getInstance().reset(editor)
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public override fun processExEntry(editor: VimEditor, context: ExecutionContext): Boolean {
|
||||||
|
val panel = ExEntryPanel.getInstance()
|
||||||
|
panel.deactivate(true)
|
||||||
|
var res = true
|
||||||
|
try {
|
||||||
|
getInstance(editor).mode = NORMAL()
|
||||||
|
|
||||||
|
logger.debug("processing command")
|
||||||
|
|
||||||
|
val text = panel.text
|
||||||
|
|
||||||
|
if (panel.label != ":") {
|
||||||
|
// Search is handled via Argument.Type.EX_STRING. Although ProcessExEntryAction is registered as the handler for
|
||||||
|
// <CR> in both command and search modes, it's only invoked for command mode (see KeyHandler.handleCommandNode).
|
||||||
|
// We should never be invoked for anything other than an actual ex command.
|
||||||
|
throw InvalidCommandException("Expected ':' command. Got '" + panel.label + "'", text)
|
||||||
|
}
|
||||||
|
|
||||||
|
logger.debug {
|
||||||
|
"swing=" + SwingUtilities.isEventDispatchThread()
|
||||||
|
}
|
||||||
|
|
||||||
|
injector.vimscriptExecutor.execute(text, editor, context, skipHistory(editor), true, CommandLineVimLContext)
|
||||||
|
} catch (e: ExException) {
|
||||||
|
VimPlugin.showMessage(e.message)
|
||||||
|
VimPlugin.indicateError()
|
||||||
|
res = false
|
||||||
|
} catch (bad: Exception) {
|
||||||
|
logger.error(bad)
|
||||||
|
VimPlugin.indicateError()
|
||||||
|
res = false
|
||||||
|
} finally {
|
||||||
|
isCommandProcessing = false
|
||||||
|
modeBeforeCommandProcessing = null
|
||||||
|
}
|
||||||
|
|
||||||
|
return res
|
||||||
|
}
|
||||||
|
|
||||||
|
// commands executed from map command / macro should not be added to history
|
||||||
|
private fun skipHistory(editor: VimEditor): Boolean {
|
||||||
|
return getInstance(editor).mappingState.isExecutingMap() || injector.macro.isExecutingMacro
|
||||||
|
}
|
||||||
|
|
||||||
|
public override fun cancelExEntry(editor: VimEditor, resetCaret: Boolean) {
|
||||||
|
editor.vimStateMachine.mode = NORMAL()
|
||||||
|
getInstance().reset(editor)
|
||||||
|
val panel = ExEntryPanel.getInstance()
|
||||||
|
panel.deactivate(true, resetCaret)
|
||||||
|
}
|
||||||
|
|
||||||
|
public override fun startFilterCommand(editor: VimEditor, context: ExecutionContext, cmd: Command) {
|
||||||
|
val initText = getRange(editor, cmd) + "!"
|
||||||
|
val currentMode = editor.mode
|
||||||
|
check(currentMode is ReturnableFromCmd) { "Cannot enable cmd mode from $currentMode" }
|
||||||
|
editor.vimStateMachine.mode = Mode.CMD_LINE(currentMode)
|
||||||
|
val panel = ExEntryPanel.getInstance()
|
||||||
|
panel.activate(editor.ij, context.ij, ":", initText, 1)
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun getRange(editor: VimEditor, cmd: Command): String {
|
||||||
|
var initText = ""
|
||||||
|
if (editor.vimStateMachine.mode is VISUAL) {
|
||||||
|
initText = "'<,'>"
|
||||||
|
} else if (cmd.rawCount > 0) {
|
||||||
|
initText = if (cmd.count == 1) {
|
||||||
|
"."
|
||||||
|
} else {
|
||||||
|
".,.+" + (cmd.count - 1)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return initText
|
||||||
|
}
|
||||||
|
|
||||||
|
@Throws(ExecutionException::class, ProcessCanceledException::class)
|
||||||
|
public override fun executeCommand(
|
||||||
|
editor: VimEditor,
|
||||||
|
command: String,
|
||||||
|
input: CharSequence?,
|
||||||
|
currentDirectoryPath: String?
|
||||||
|
): String? {
|
||||||
|
// This is a much simplified version of how Vim does this. We're using stdin/stdout directly, while Vim will
|
||||||
|
// redirect to temp files ('shellredir' and 'shelltemp') or use pipes. We don't support 'shellquote', because we're
|
||||||
|
// not handling redirection, but we do use 'shellxquote' and 'shellxescape', because these have defaults that work
|
||||||
|
// better with Windows. We also don't bother using ShellExecute for Windows commands beginning with `start`.
|
||||||
|
// Finally, we're also not bothering with the crazy space and backslash handling of the 'shell' options content.
|
||||||
|
|
||||||
|
return ProgressManager.getInstance().runProcessWithProgressSynchronously<String, ExecutionException>(
|
||||||
|
{
|
||||||
|
val shell = injector.globalOptions().shell
|
||||||
|
val shellcmdflag = injector.globalOptions().shellcmdflag
|
||||||
|
val shellxescape = injector.globalOptions().shellxescape
|
||||||
|
val shellxquote = injector.globalOptions().shellxquote
|
||||||
|
|
||||||
|
// For Win32. See :help 'shellxescape'
|
||||||
|
val escapedCommand = if (shellxquote == "(") doEscape(command, shellxescape, "^")
|
||||||
|
else command
|
||||||
|
// Required for Win32+cmd.exe, defaults to "(". See :help 'shellxquote'
|
||||||
|
val quotedCommand = if (shellxquote == "(") "($escapedCommand)"
|
||||||
|
else (if (shellxquote == "\"(") "\"($escapedCommand)\""
|
||||||
|
else shellxquote + escapedCommand + shellxquote)
|
||||||
|
|
||||||
|
val commands = ArrayList<String>()
|
||||||
|
commands.add(shell)
|
||||||
|
if (shellcmdflag.isNotEmpty()) {
|
||||||
|
// Note that Vim also does a simple whitespace split for multiple parameters
|
||||||
|
commands.addAll(ParametersListUtil.parse(shellcmdflag))
|
||||||
|
}
|
||||||
|
commands.add(quotedCommand)
|
||||||
|
|
||||||
|
if (logger.isDebugEnabled) {
|
||||||
|
logger.debug(String.format("shell=%s shellcmdflag=%s command=%s", shell, shellcmdflag, quotedCommand))
|
||||||
|
}
|
||||||
|
|
||||||
|
val commandLine = GeneralCommandLine(commands)
|
||||||
|
if (currentDirectoryPath != null) {
|
||||||
|
commandLine.setWorkDirectory(currentDirectoryPath)
|
||||||
|
}
|
||||||
|
val handler = CapturingProcessHandler(commandLine)
|
||||||
|
if (input != null) {
|
||||||
|
handler.addProcessListener(object : ProcessAdapter() {
|
||||||
|
override fun startNotified(event: ProcessEvent) {
|
||||||
|
try {
|
||||||
|
val charSequenceReader = CharSequenceReader(input)
|
||||||
|
val outputStreamWriter = BufferedWriter(OutputStreamWriter(handler.processInput))
|
||||||
|
copy(charSequenceReader, outputStreamWriter)
|
||||||
|
outputStreamWriter.close()
|
||||||
|
} catch (e: IOException) {
|
||||||
|
logger.error(e)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
val progressIndicator = ProgressIndicatorProvider.getInstance().progressIndicator
|
||||||
|
val output = handler.runProcessWithProgressIndicator(progressIndicator)
|
||||||
|
|
||||||
|
lastCommand = command
|
||||||
|
|
||||||
|
if (output.isCancelled) {
|
||||||
|
// TODO: Vim will use whatever text has already been written to stdout
|
||||||
|
// For whatever reason, we're not getting any here, so just throw an exception
|
||||||
|
throw ProcessCanceledException()
|
||||||
|
}
|
||||||
|
|
||||||
|
val exitCode = handler.exitCode
|
||||||
|
if (exitCode != null && exitCode != 0) {
|
||||||
|
VimPlugin.showMessage("shell returned $exitCode")
|
||||||
|
VimPlugin.indicateError()
|
||||||
|
}
|
||||||
|
(output.stderr + output.stdout).replace("\u001B\\[[;\\d]*m".toRegex(), "")
|
||||||
|
}, "IdeaVim - !$command", true, editor.ij.project
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
@Suppress("SameParameterValue")
|
||||||
|
private fun doEscape(original: String, charsToEscape: String, escapeChar: String): String {
|
||||||
|
var result = original
|
||||||
|
for (c in charsToEscape.toCharArray()) {
|
||||||
|
result = result.replace("" + c, escapeChar + c)
|
||||||
|
}
|
||||||
|
return result
|
||||||
|
}
|
||||||
|
|
||||||
|
// TODO: Java 10 has a transferTo method we could use instead
|
||||||
|
@Throws(IOException::class)
|
||||||
|
private fun copy(from: Reader, to: Writer) {
|
||||||
|
val buf = CharArray(2048)
|
||||||
|
var cnt: Int
|
||||||
|
while ((from.read(buf).also { cnt = it }) != -1) {
|
||||||
|
to.write(buf, 0, cnt)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public companion object {
|
||||||
|
private val logger = logger<ProcessGroup>()
|
||||||
|
}
|
||||||
|
}
|
@@ -36,10 +36,9 @@ import com.maddyhome.idea.vim.history.HistoryConstants;
|
|||||||
import com.maddyhome.idea.vim.newapi.IjEditorExecutionContext;
|
import com.maddyhome.idea.vim.newapi.IjEditorExecutionContext;
|
||||||
import com.maddyhome.idea.vim.newapi.IjVimCaret;
|
import com.maddyhome.idea.vim.newapi.IjVimCaret;
|
||||||
import com.maddyhome.idea.vim.newapi.IjVimEditor;
|
import com.maddyhome.idea.vim.newapi.IjVimEditor;
|
||||||
|
import com.maddyhome.idea.vim.newapi.IjVimSearchGroup;
|
||||||
import com.maddyhome.idea.vim.options.GlobalOptionChangeListener;
|
import com.maddyhome.idea.vim.options.GlobalOptionChangeListener;
|
||||||
import com.maddyhome.idea.vim.regexp.CharPointer;
|
import com.maddyhome.idea.vim.regexp.*;
|
||||||
import com.maddyhome.idea.vim.regexp.CharacterClasses;
|
|
||||||
import com.maddyhome.idea.vim.regexp.RegExp;
|
|
||||||
import com.maddyhome.idea.vim.ui.ModalEntry;
|
import com.maddyhome.idea.vim.ui.ModalEntry;
|
||||||
import com.maddyhome.idea.vim.ui.ex.ExEntryPanel;
|
import com.maddyhome.idea.vim.ui.ex.ExEntryPanel;
|
||||||
import com.maddyhome.idea.vim.vimscript.model.VimLContext;
|
import com.maddyhome.idea.vim.vimscript.model.VimLContext;
|
||||||
@@ -62,34 +61,46 @@ import java.util.*;
|
|||||||
import static com.maddyhome.idea.vim.api.VimInjectorKt.*;
|
import static com.maddyhome.idea.vim.api.VimInjectorKt.*;
|
||||||
import static com.maddyhome.idea.vim.helper.HelperKt.localEditors;
|
import static com.maddyhome.idea.vim.helper.HelperKt.localEditors;
|
||||||
import static com.maddyhome.idea.vim.helper.SearchHelperKtKt.shouldIgnoreCase;
|
import static com.maddyhome.idea.vim.helper.SearchHelperKtKt.shouldIgnoreCase;
|
||||||
|
import static com.maddyhome.idea.vim.newapi.IjVimInjectorKt.globalIjOptions;
|
||||||
import static com.maddyhome.idea.vim.register.RegisterConstants.LAST_SEARCH_REGISTER;
|
import static com.maddyhome.idea.vim.register.RegisterConstants.LAST_SEARCH_REGISTER;
|
||||||
|
|
||||||
@State(name = "VimSearchSettings", storages = {
|
@State(name = "VimSearchSettings", storages = {
|
||||||
@Storage(value = "$APP_CONFIG$/vim_settings_local.xml", roamingType = RoamingType.DISABLED)
|
@Storage(value = "$APP_CONFIG$/vim_settings_local.xml", roamingType = RoamingType.DISABLED)
|
||||||
})
|
})
|
||||||
public class SearchGroup extends VimSearchGroupBase implements PersistentStateComponent<Element> {
|
@Deprecated
|
||||||
|
/**
|
||||||
|
* @deprecated Replace with IjVimSearchGroup
|
||||||
|
*/
|
||||||
|
public class SearchGroup extends IjVimSearchGroup implements PersistentStateComponent<Element> {
|
||||||
public SearchGroup() {
|
public SearchGroup() {
|
||||||
// TODO: Investigate migrating these listeners to use the effective value change listener
|
super();
|
||||||
// This would allow us to update the editor we're told to update, rather than looping over all projects and updating
|
if (!globalIjOptions(injector).getUseNewRegex()) {
|
||||||
// the highlights in that project's current document's open editors (see VIM-2779).
|
// TODO: Investigate migrating these listeners to use the effective value change listener
|
||||||
// However, we probably only want to update the editors associated with the current document, so maybe the whole
|
// This would allow us to update the editor we're told to update, rather than looping over all projects and updating
|
||||||
// code needs to be reworked. We're currently using the same update code for changes in the search term as well as
|
// the highlights in that project's current document's open editors (see VIM-2779).
|
||||||
// changes in the search options.
|
// However, we probably only want to update the editors associated with the current document, so maybe the whole
|
||||||
VimPlugin.getOptionGroup().addGlobalOptionChangeListener(Options.hlsearch, () -> {
|
// code needs to be reworked. We're currently using the same update code for changes in the search term as well as
|
||||||
resetShowSearchHighlight();
|
// changes in the search options.
|
||||||
forceUpdateSearchHighlights();
|
VimPlugin.getOptionGroup().addGlobalOptionChangeListener(Options.hlsearch, () -> {
|
||||||
});
|
resetShowSearchHighlight();
|
||||||
|
|
||||||
final GlobalOptionChangeListener updateHighlightsIfVisible = () -> {
|
|
||||||
if (showSearchHighlight) {
|
|
||||||
forceUpdateSearchHighlights();
|
forceUpdateSearchHighlights();
|
||||||
}
|
});
|
||||||
};
|
|
||||||
VimPlugin.getOptionGroup().addGlobalOptionChangeListener(Options.ignorecase, updateHighlightsIfVisible);
|
final GlobalOptionChangeListener updateHighlightsIfVisible = () -> {
|
||||||
VimPlugin.getOptionGroup().addGlobalOptionChangeListener(Options.smartcase, updateHighlightsIfVisible);
|
if (showSearchHighlight) {
|
||||||
|
forceUpdateSearchHighlights();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
VimPlugin.getOptionGroup().addGlobalOptionChangeListener(Options.ignorecase, updateHighlightsIfVisible);
|
||||||
|
VimPlugin.getOptionGroup().addGlobalOptionChangeListener(Options.smartcase, updateHighlightsIfVisible);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public void turnOn() {
|
public void turnOn() {
|
||||||
|
if (globalIjOptions(injector).getUseNewRegex()) {
|
||||||
|
super.updateSearchHighlights(false);
|
||||||
|
return;
|
||||||
|
}
|
||||||
updateSearchHighlights();
|
updateSearchHighlights();
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -100,7 +111,12 @@ public class SearchGroup extends VimSearchGroupBase implements PersistentStateCo
|
|||||||
}
|
}
|
||||||
|
|
||||||
@TestOnly
|
@TestOnly
|
||||||
|
@Override
|
||||||
public void resetState() {
|
public void resetState() {
|
||||||
|
if (globalIjOptions(injector).getUseNewRegex()) {
|
||||||
|
super.resetState();
|
||||||
|
return;
|
||||||
|
}
|
||||||
lastPatternIdx = RE_SEARCH;
|
lastPatternIdx = RE_SEARCH;
|
||||||
lastSearch = lastSubstitute = lastReplace = null;
|
lastSearch = lastSubstitute = lastReplace = null;
|
||||||
lastPatternOffset = "";
|
lastPatternOffset = "";
|
||||||
@@ -114,7 +130,9 @@ public class SearchGroup extends VimSearchGroupBase implements PersistentStateCo
|
|||||||
*
|
*
|
||||||
* @return The pattern used for last search. Can be null
|
* @return The pattern used for last search. Can be null
|
||||||
*/
|
*/
|
||||||
|
@Override
|
||||||
public @Nullable String getLastSearchPattern() {
|
public @Nullable String getLastSearchPattern() {
|
||||||
|
if (globalIjOptions(injector).getUseNewRegex()) return super.getLastSearchPattern();
|
||||||
return lastSearch;
|
return lastSearch;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -122,7 +140,9 @@ public class SearchGroup extends VimSearchGroupBase implements PersistentStateCo
|
|||||||
* Get the last pattern used in substitution.
|
* Get the last pattern used in substitution.
|
||||||
* @return The pattern used for the last substitute command. Can be null
|
* @return The pattern used for the last substitute command. Can be null
|
||||||
*/
|
*/
|
||||||
|
@Override
|
||||||
public @Nullable String getLastSubstitutePattern() {
|
public @Nullable String getLastSubstitutePattern() {
|
||||||
|
if (globalIjOptions(injector).getUseNewRegex()) return super.getLastSubstitutePattern();
|
||||||
return lastSubstitute;
|
return lastSubstitute;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -131,7 +151,9 @@ public class SearchGroup extends VimSearchGroupBase implements PersistentStateCo
|
|||||||
*
|
*
|
||||||
* @return The pattern last used for either searching or substitution. Can be null
|
* @return The pattern last used for either searching or substitution. Can be null
|
||||||
*/
|
*/
|
||||||
public @Nullable String getLastUsedPattern() {
|
@Override
|
||||||
|
protected @Nullable String getLastUsedPattern() {
|
||||||
|
if (globalIjOptions(injector).getUseNewRegex()) return super.getLastUsedPattern();
|
||||||
switch (lastPatternIdx) {
|
switch (lastPatternIdx) {
|
||||||
case RE_SEARCH: return lastSearch;
|
case RE_SEARCH: return lastSearch;
|
||||||
case RE_SUBST: return lastSubstitute;
|
case RE_SUBST: return lastSubstitute;
|
||||||
@@ -195,6 +217,10 @@ public class SearchGroup extends VimSearchGroupBase implements PersistentStateCo
|
|||||||
@TestOnly
|
@TestOnly
|
||||||
public void setLastSearchState(@SuppressWarnings("unused") @NotNull Editor editor, @NotNull String pattern,
|
public void setLastSearchState(@SuppressWarnings("unused") @NotNull Editor editor, @NotNull String pattern,
|
||||||
@NotNull String patternOffset, Direction direction) {
|
@NotNull String patternOffset, Direction direction) {
|
||||||
|
if (globalIjOptions(injector).getUseNewRegex()) {
|
||||||
|
super.setLastSearchState(pattern, patternOffset, direction);
|
||||||
|
return;
|
||||||
|
}
|
||||||
setLastUsedPattern(pattern, RE_SEARCH, true);
|
setLastUsedPattern(pattern, RE_SEARCH, true);
|
||||||
lastIgnoreSmartCase = false;
|
lastIgnoreSmartCase = false;
|
||||||
lastPatternOffset = patternOffset;
|
lastPatternOffset = patternOffset;
|
||||||
@@ -226,7 +252,7 @@ public class SearchGroup extends VimSearchGroupBase implements PersistentStateCo
|
|||||||
int startLine,
|
int startLine,
|
||||||
int endLine,
|
int endLine,
|
||||||
boolean ignoreCase) {
|
boolean ignoreCase) {
|
||||||
return SearchHelper.findAll(editor, pattern, startLine, endLine, ignoreCase);
|
return injector.getSearchHelper().findAll(new IjVimEditor(editor), pattern, startLine, endLine, ignoreCase);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -254,6 +280,8 @@ public class SearchGroup extends VimSearchGroupBase implements PersistentStateCo
|
|||||||
*/
|
*/
|
||||||
@Override
|
@Override
|
||||||
public int processSearchCommand(@NotNull VimEditor editor, @NotNull String command, int startOffset, @NotNull Direction dir) {
|
public int processSearchCommand(@NotNull VimEditor editor, @NotNull String command, int startOffset, @NotNull Direction dir) {
|
||||||
|
if (globalIjOptions(injector).getUseNewRegex()) return super.processSearchCommand(editor, command, startOffset, dir);
|
||||||
|
|
||||||
boolean isNewPattern = false;
|
boolean isNewPattern = false;
|
||||||
String pattern = null;
|
String pattern = null;
|
||||||
String patternOffset = null;
|
String patternOffset = null;
|
||||||
@@ -414,6 +442,7 @@ public class SearchGroup extends VimSearchGroupBase implements PersistentStateCo
|
|||||||
*/
|
*/
|
||||||
@Override
|
@Override
|
||||||
public int searchWord(@NotNull VimEditor editor, @NotNull ImmutableVimCaret caret, int count, boolean whole, @NotNull Direction dir) {
|
public int searchWord(@NotNull VimEditor editor, @NotNull ImmutableVimCaret caret, int count, boolean whole, @NotNull Direction dir) {
|
||||||
|
if (globalIjOptions(injector).getUseNewRegex()) return super.searchWord(editor, caret, count, whole, dir);
|
||||||
TextRange range = SearchHelper.findWordUnderCursor(((IjVimEditor)editor).getEditor(), ((IjVimCaret)caret).getCaret());
|
TextRange range = SearchHelper.findWordUnderCursor(((IjVimEditor)editor).getEditor(), ((IjVimCaret)caret).getCaret());
|
||||||
if (range == null) {
|
if (range == null) {
|
||||||
logger.warn("No range was found");
|
logger.warn("No range was found");
|
||||||
@@ -455,6 +484,7 @@ public class SearchGroup extends VimSearchGroupBase implements PersistentStateCo
|
|||||||
*/
|
*/
|
||||||
@Override
|
@Override
|
||||||
public int searchNext(@NotNull VimEditor editor, @NotNull ImmutableVimCaret caret, int count) {
|
public int searchNext(@NotNull VimEditor editor, @NotNull ImmutableVimCaret caret, int count) {
|
||||||
|
if (globalIjOptions(injector).getUseNewRegex()) return super.searchNext(editor, caret, count);
|
||||||
return searchNextWithDirection(((IjVimEditor)editor).getEditor(), ((IjVimCaret)caret).getCaret(), count, lastDir);
|
return searchNextWithDirection(((IjVimEditor)editor).getEditor(), ((IjVimCaret)caret).getCaret(), count, lastDir);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -471,6 +501,7 @@ public class SearchGroup extends VimSearchGroupBase implements PersistentStateCo
|
|||||||
*/
|
*/
|
||||||
@Override
|
@Override
|
||||||
public int searchPrevious(@NotNull VimEditor editor, @NotNull ImmutableVimCaret caret, int count) {
|
public int searchPrevious(@NotNull VimEditor editor, @NotNull ImmutableVimCaret caret, int count) {
|
||||||
|
if (globalIjOptions(injector).getUseNewRegex()) return super.searchPrevious(editor, caret, count);
|
||||||
return searchNextWithDirection(((IjVimEditor)editor).getEditor(), ((IjVimCaret)caret).getCaret(), count,
|
return searchNextWithDirection(((IjVimEditor)editor).getEditor(), ((IjVimCaret)caret).getCaret(), count,
|
||||||
lastDir.reverse());
|
lastDir.reverse());
|
||||||
}
|
}
|
||||||
@@ -524,6 +555,8 @@ public class SearchGroup extends VimSearchGroupBase implements PersistentStateCo
|
|||||||
@NotNull @NonNls String excmd,
|
@NotNull @NonNls String excmd,
|
||||||
@NotNull @NonNls String exarg,
|
@NotNull @NonNls String exarg,
|
||||||
@NotNull VimLContext parent) {
|
@NotNull VimLContext parent) {
|
||||||
|
if (globalIjOptions(injector).getUseNewRegex()) return super.processSubstituteCommand(editor, caret, range, excmd, exarg, parent);
|
||||||
|
|
||||||
// Explicitly exit visual mode here, so that visual mode marks don't change when we move the cursor to a match.
|
// Explicitly exit visual mode here, so that visual mode marks don't change when we move the cursor to a match.
|
||||||
List<ExException> exceptions = new ArrayList<>();
|
List<ExException> exceptions = new ArrayList<>();
|
||||||
if (CommandStateHelper.inVisualMode(((IjVimEditor) editor).getEditor())) {
|
if (CommandStateHelper.inVisualMode(((IjVimEditor) editor).getEditor())) {
|
||||||
@@ -687,8 +720,7 @@ public class SearchGroup extends VimSearchGroupBase implements PersistentStateCo
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
Pair<Boolean, Triple<Object, String, Object>> booleanregmmatch_tPair = search_regcomp(pat, which_pat,
|
Pair<Boolean, Triple<Object, String, Object>> booleanregmmatch_tPair = search_regcomp(pat, which_pat, RE_SUBST);
|
||||||
RE_SUBST);
|
|
||||||
if (!booleanregmmatch_tPair.getFirst()) {
|
if (!booleanregmmatch_tPair.getFirst()) {
|
||||||
if (do_error) {
|
if (do_error) {
|
||||||
VimPlugin.showMessage(MessageHelper.message(Msg.e_invcmd));
|
VimPlugin.showMessage(MessageHelper.message(Msg.e_invcmd));
|
||||||
@@ -696,9 +728,9 @@ public class SearchGroup extends VimSearchGroupBase implements PersistentStateCo
|
|||||||
}
|
}
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
RegExp.regmmatch_T regmatch = (RegExp.regmmatch_T) booleanregmmatch_tPair.getSecond().getFirst();
|
RegExp.regmmatch_T regmatch = (RegExp.regmmatch_T)booleanregmmatch_tPair.getSecond().getFirst();
|
||||||
String pattern = booleanregmmatch_tPair.getSecond().getSecond();
|
String pattern = booleanregmmatch_tPair.getSecond().getSecond();
|
||||||
RegExp sp = (RegExp) booleanregmmatch_tPair.getSecond().getThird();
|
RegExp sp = (RegExp)booleanregmmatch_tPair.getSecond().getThird();
|
||||||
|
|
||||||
/* the 'i' or 'I' flag overrules 'ignorecase' and 'smartcase' */
|
/* the 'i' or 'I' flag overrules 'ignorecase' and 'smartcase' */
|
||||||
if (do_ic == 'i') {
|
if (do_ic == 'i') {
|
||||||
@@ -731,8 +763,8 @@ public class SearchGroup extends VimSearchGroupBase implements PersistentStateCo
|
|||||||
resetShowSearchHighlight();
|
resetShowSearchHighlight();
|
||||||
forceUpdateSearchHighlights();
|
forceUpdateSearchHighlights();
|
||||||
|
|
||||||
int start = ((IjVimEditor) editor).getEditor().getDocument().getLineStartOffset(line1);
|
int start = ((IjVimEditor)editor).getEditor().getDocument().getLineStartOffset(line1);
|
||||||
int end = ((IjVimEditor) editor).getEditor().getDocument().getLineEndOffset(line2);
|
int end = ((IjVimEditor)editor).getEditor().getDocument().getLineEndOffset(line2);
|
||||||
|
|
||||||
if (logger.isDebugEnabled()) {
|
if (logger.isDebugEnabled()) {
|
||||||
logger.debug("search range=[" + start + "," + end + "]");
|
logger.debug("search range=[" + start + "," + end + "]");
|
||||||
@@ -755,7 +787,6 @@ public class SearchGroup extends VimSearchGroupBase implements PersistentStateCo
|
|||||||
firstMatch = false;
|
firstMatch = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
String match = sp.vim_regsub_multi(regmatch, lnum, sub, 1, false);
|
String match = sp.vim_regsub_multi(regmatch, lnum, sub, 1, false);
|
||||||
if (sub.charAt(0) == '\\' && sub.charAt(1) == '=') {
|
if (sub.charAt(0) == '\\' && sub.charAt(1) == '=') {
|
||||||
String exprString = sub.toString().substring(2);
|
String exprString = sub.toString().substring(2);
|
||||||
@@ -764,22 +795,25 @@ public class SearchGroup extends VimSearchGroupBase implements PersistentStateCo
|
|||||||
exceptions.add(new ExException("E15: Invalid expression: " + exprString));
|
exceptions.add(new ExException("E15: Invalid expression: " + exprString));
|
||||||
expression = new SimpleExpression(new VimString(""));
|
expression = new SimpleExpression(new VimString(""));
|
||||||
}
|
}
|
||||||
} else if (match == null) {
|
}
|
||||||
|
else if (match == null) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
int line = lnum + regmatch.startpos[0].lnum;
|
int line = lnum + regmatch.startpos[0].lnum;
|
||||||
CharacterPosition startpos = new CharacterPosition(lnum + regmatch.startpos[0].lnum, regmatch.startpos[0].col);
|
CharacterPosition startpos = new CharacterPosition(lnum + regmatch.startpos[0].lnum, regmatch.startpos[0].col);
|
||||||
CharacterPosition endpos = new CharacterPosition(lnum + regmatch.endpos[0].lnum, regmatch.endpos[0].col);
|
CharacterPosition endpos = new CharacterPosition(lnum + regmatch.endpos[0].lnum, regmatch.endpos[0].col);
|
||||||
int startoff = startpos.toOffset(((IjVimEditor) editor).getEditor());
|
int startoff = startpos.toOffset(((IjVimEditor)editor).getEditor());
|
||||||
int endoff = endpos.toOffset(((IjVimEditor) editor).getEditor());
|
int endoff = endpos.toOffset(((IjVimEditor)editor).getEditor());
|
||||||
|
|
||||||
if (do_all || line != lastLine) {
|
if (do_all || line != lastLine) {
|
||||||
boolean doReplace = true;
|
boolean doReplace = true;
|
||||||
if (do_ask) {
|
if (do_ask) {
|
||||||
RangeHighlighter hl = SearchHighlightsHelper.addSubstitutionConfirmationHighlight(((IjVimEditor) editor).getEditor(), startoff, endoff);
|
RangeHighlighter hl =
|
||||||
final ReplaceConfirmationChoice choice = confirmChoice(((IjVimEditor) editor).getEditor(), match, ((IjVimCaret) caret).getCaret(), startoff);
|
SearchHighlightsHelper.addSubstitutionConfirmationHighlight(((IjVimEditor)editor).getEditor(), startoff,
|
||||||
((IjVimEditor) editor).getEditor().getMarkupModel().removeHighlighter(hl);
|
endoff);
|
||||||
|
final ReplaceConfirmationChoice choice = confirmChoice(((IjVimEditor)editor).getEditor(), match, ((IjVimCaret)caret).getCaret(), startoff);
|
||||||
|
((IjVimEditor)editor).getEditor().getMarkupModel().removeHighlighter(hl);
|
||||||
switch (choice) {
|
switch (choice) {
|
||||||
case SUBSTITUTE_THIS:
|
case SUBSTITUTE_THIS:
|
||||||
doReplace = true;
|
doReplace = true;
|
||||||
@@ -802,25 +836,26 @@ public class SearchGroup extends VimSearchGroupBase implements PersistentStateCo
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (doReplace) {
|
if (doReplace) {
|
||||||
SubmatchFunctionHandler.Companion.getInstance().setLatestMatch(((IjVimEditor) editor).getEditor().getDocument().getText(new com.intellij.openapi.util.TextRange(startoff, endoff)));
|
SubmatchFunctionHandler.Companion.getInstance().setLatestMatch(
|
||||||
|
((IjVimEditor)editor).getEditor().getDocument().getText(new com.intellij.openapi.util.TextRange(startoff, endoff)));
|
||||||
caret.moveToOffset(startoff);
|
caret.moveToOffset(startoff);
|
||||||
if (expression != null) {
|
if (expression != null) {
|
||||||
try {
|
try {
|
||||||
match = expression
|
match =
|
||||||
.evaluate(editor, injector.getExecutionContextManager().onEditor(editor, null), parent)
|
expression.evaluate(editor, injector.getExecutionContextManager().onEditor(editor, null), parent).toInsertableString();
|
||||||
.toInsertableString();
|
}
|
||||||
} catch (Exception e) {
|
catch (Exception e) {
|
||||||
exceptions.add((ExException) e);
|
exceptions.add((ExException)e);
|
||||||
match = "";
|
match = "";
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
String finalMatch = match;
|
String finalMatch = match;
|
||||||
ApplicationManager.getApplication().runWriteAction(() -> ((IjVimEditor) editor).getEditor().getDocument().replaceString(startoff, endoff,
|
ApplicationManager.getApplication().runWriteAction(
|
||||||
finalMatch));
|
() -> ((IjVimEditor)editor).getEditor().getDocument().replaceString(startoff, endoff, finalMatch));
|
||||||
lastMatch = startoff;
|
lastMatch = startoff;
|
||||||
int newend = startoff + match.length();
|
int newend = startoff + match.length();
|
||||||
newpos = CharacterPosition.Companion.fromOffset(((IjVimEditor) editor).getEditor(), newend);
|
newpos = CharacterPosition.Companion.fromOffset(((IjVimEditor)editor).getEditor(), newend);
|
||||||
|
|
||||||
lnum += newpos.line - endpos.line;
|
lnum += newpos.line - endpos.line;
|
||||||
line2 += newpos.line - endpos.line;
|
line2 += newpos.line - endpos.line;
|
||||||
@@ -875,6 +910,10 @@ public class SearchGroup extends VimSearchGroupBase implements PersistentStateCo
|
|||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void setLastSearchPattern(@Nullable String lastSearchPattern) {
|
public void setLastSearchPattern(@Nullable String lastSearchPattern) {
|
||||||
|
if (globalIjOptions(injector).getUseNewRegex()) {
|
||||||
|
super.setLastSearchPattern(lastSearchPattern);
|
||||||
|
return;
|
||||||
|
}
|
||||||
this.lastSearch = lastSearchPattern;
|
this.lastSearch = lastSearchPattern;
|
||||||
if (showSearchHighlight) {
|
if (showSearchHighlight) {
|
||||||
resetIncsearchHighlights();
|
resetIncsearchHighlights();
|
||||||
@@ -884,6 +923,10 @@ public class SearchGroup extends VimSearchGroupBase implements PersistentStateCo
|
|||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void setLastSubstitutePattern(@Nullable String lastSubstitutePattern) {
|
public void setLastSubstitutePattern(@Nullable String lastSubstitutePattern) {
|
||||||
|
if (globalIjOptions(injector).getUseNewRegex()) {
|
||||||
|
super.setLastSubstitutePattern(lastSubstitutePattern);
|
||||||
|
return;
|
||||||
|
}
|
||||||
this.lastSubstitute = lastSubstitutePattern;
|
this.lastSubstitute = lastSubstitutePattern;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -893,6 +936,7 @@ public class SearchGroup extends VimSearchGroupBase implements PersistentStateCo
|
|||||||
int patternOffset,
|
int patternOffset,
|
||||||
int startOffset,
|
int startOffset,
|
||||||
@NotNull Direction direction) {
|
@NotNull Direction direction) {
|
||||||
|
if (globalIjOptions(injector).getUseNewRegex()) return super.processSearchRange(editor, pattern, patternOffset, startOffset, direction);
|
||||||
return processSearchRange(((IjVimEditor) editor).getEditor(), pattern, patternOffset, startOffset, direction);
|
return processSearchRange(((IjVimEditor) editor).getEditor(), pattern, patternOffset, startOffset, direction);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -1015,6 +1059,7 @@ public class SearchGroup extends VimSearchGroupBase implements PersistentStateCo
|
|||||||
*/
|
*/
|
||||||
@Override
|
@Override
|
||||||
public @Nullable TextRange getNextSearchRange(@NotNull VimEditor editor, int count, boolean forwards) {
|
public @Nullable TextRange getNextSearchRange(@NotNull VimEditor editor, int count, boolean forwards) {
|
||||||
|
if (globalIjOptions(injector).getUseNewRegex()) return super.getNextSearchRange(editor, count, forwards);
|
||||||
editor.removeSecondaryCarets();
|
editor.removeSecondaryCarets();
|
||||||
TextRange current = findUnderCaret(editor);
|
TextRange current = findUnderCaret(editor);
|
||||||
|
|
||||||
@@ -1046,17 +1091,10 @@ public class SearchGroup extends VimSearchGroupBase implements PersistentStateCo
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
|
||||||
@Nullable
|
|
||||||
public TextRange findUnderCaret(@NotNull VimEditor editor) {
|
|
||||||
final TextRange backSearch = searchBackward(editor, editor.primaryCaret().getOffset().getPoint() + 1, 1);
|
|
||||||
if (backSearch == null) return null;
|
|
||||||
return backSearch.contains(editor.primaryCaret().getOffset().getPoint()) ? backSearch : null;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
@Nullable
|
@Nullable
|
||||||
public TextRange searchBackward(@NotNull VimEditor editor, int offset, int count) {
|
public TextRange searchBackward(@NotNull VimEditor editor, int offset, int count) {
|
||||||
|
if (globalIjOptions(injector).getUseNewRegex()) return super.searchBackward(editor, offset, count);
|
||||||
// Backward search returns wrongs end offset for some cases. That's why we should perform additional forward search
|
// Backward search returns wrongs end offset for some cases. That's why we should perform additional forward search
|
||||||
final EnumSet<SearchOptions> searchOptions = EnumSet.of(SearchOptions.WRAP, SearchOptions.WHOLE_FILE, SearchOptions.BACKWARDS);
|
final EnumSet<SearchOptions> searchOptions = EnumSet.of(SearchOptions.WRAP, SearchOptions.WHOLE_FILE, SearchOptions.BACKWARDS);
|
||||||
final TextRange foundBackward = VimInjectorKt.getInjector().getSearchHelper().findPattern(editor, getLastUsedPattern(), offset, count, searchOptions);
|
final TextRange foundBackward = VimInjectorKt.getInjector().getSearchHelper().findPattern(editor, getLastUsedPattern(), offset, count, searchOptions);
|
||||||
@@ -1074,7 +1112,12 @@ public class SearchGroup extends VimSearchGroupBase implements PersistentStateCo
|
|||||||
//
|
//
|
||||||
// *******************************************************************************************************************
|
// *******************************************************************************************************************
|
||||||
//region Search highlights
|
//region Search highlights
|
||||||
|
@Override
|
||||||
public void clearSearchHighlight() {
|
public void clearSearchHighlight() {
|
||||||
|
if (globalIjOptions(injector).getUseNewRegex()) {
|
||||||
|
super.clearSearchHighlight();
|
||||||
|
return;
|
||||||
|
}
|
||||||
showSearchHighlight = false;
|
showSearchHighlight = false;
|
||||||
updateSearchHighlights();
|
updateSearchHighlights();
|
||||||
}
|
}
|
||||||
@@ -1094,7 +1137,12 @@ public class SearchGroup extends VimSearchGroupBase implements PersistentStateCo
|
|||||||
/**
|
/**
|
||||||
* Reset the search highlights to the last used pattern after highlighting incsearch results.
|
* Reset the search highlights to the last used pattern after highlighting incsearch results.
|
||||||
*/
|
*/
|
||||||
|
@Override
|
||||||
public void resetIncsearchHighlights() {
|
public void resetIncsearchHighlights() {
|
||||||
|
if (globalIjOptions(injector).getUseNewRegex()) {
|
||||||
|
super.resetIncsearchHighlights();
|
||||||
|
return;
|
||||||
|
}
|
||||||
SearchHighlightsHelper.updateSearchHighlights(getLastUsedPattern(), lastIgnoreSmartCase, showSearchHighlight, true);
|
SearchHighlightsHelper.updateSearchHighlights(getLastUsedPattern(), lastIgnoreSmartCase, showSearchHighlight, true);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -1103,9 +1151,13 @@ public class SearchGroup extends VimSearchGroupBase implements PersistentStateCo
|
|||||||
}
|
}
|
||||||
|
|
||||||
private void highlightSearchLines(@NotNull Editor editor, int startLine, int endLine) {
|
private void highlightSearchLines(@NotNull Editor editor, int startLine, int endLine) {
|
||||||
|
if (globalIjOptions(injector).getUseNewRegex()) {
|
||||||
|
super.highlightSearchLines(new IjVimEditor(editor), startLine, endLine);
|
||||||
|
return;
|
||||||
|
}
|
||||||
final String pattern = getLastUsedPattern();
|
final String pattern = getLastUsedPattern();
|
||||||
if (pattern != null) {
|
if (pattern != null) {
|
||||||
final List<TextRange> results = SearchHelper.findAll(editor, pattern, startLine, endLine,
|
final List<TextRange> results = injector.getSearchHelper().findAll(new IjVimEditor(editor), pattern, startLine, endLine,
|
||||||
shouldIgnoreCase(pattern, lastIgnoreSmartCase));
|
shouldIgnoreCase(pattern, lastIgnoreSmartCase));
|
||||||
SearchHighlightsHelper.highlightSearchResults(editor, pattern, results, -1);
|
SearchHighlightsHelper.highlightSearchResults(editor, pattern, results, -1);
|
||||||
}
|
}
|
||||||
@@ -1114,12 +1166,17 @@ public class SearchGroup extends VimSearchGroupBase implements PersistentStateCo
|
|||||||
/**
|
/**
|
||||||
* Updates search highlights when the selected editor changes
|
* Updates search highlights when the selected editor changes
|
||||||
*/
|
*/
|
||||||
public static void fileEditorManagerSelectionChangedCallback(@SuppressWarnings("unused") @NotNull FileEditorManagerEvent event) {
|
public void fileEditorManagerSelectionChangedCallback(@SuppressWarnings("unused") @NotNull FileEditorManagerEvent event) {
|
||||||
|
if (globalIjOptions(injector).getUseNewRegex()) {
|
||||||
|
super.updateSearchHighlights(false);
|
||||||
|
return;
|
||||||
|
}
|
||||||
VimPlugin.getSearch().updateSearchHighlights();
|
VimPlugin.getSearch().updateSearchHighlights();
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public Integer findDecimalNumber(@NotNull String line) {
|
public Integer findDecimalNumber(@NotNull String line) {
|
||||||
|
if (globalIjOptions(injector).getUseNewRegex()) return super.findDecimalNumber(line);
|
||||||
Pair<TextRange, NumberType> searchResult = SearchHelper.findNumberInText(line, 0, false, false, false);
|
Pair<TextRange, NumberType> searchResult = SearchHelper.findNumberInText(line, 0, false, false, false);
|
||||||
if (searchResult != null) {
|
if (searchResult != null) {
|
||||||
TextRange range = searchResult.component1();
|
TextRange range = searchResult.component1();
|
||||||
@@ -1131,6 +1188,7 @@ public class SearchGroup extends VimSearchGroupBase implements PersistentStateCo
|
|||||||
@NotNull
|
@NotNull
|
||||||
@Override
|
@Override
|
||||||
public Direction getLastSearchDirection() {
|
public Direction getLastSearchDirection() {
|
||||||
|
if (globalIjOptions(injector).getUseNewRegex()) return super.getLastSearchDirection();
|
||||||
return lastDir;
|
return lastDir;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -1274,7 +1332,7 @@ public class SearchGroup extends VimSearchGroupBase implements PersistentStateCo
|
|||||||
if (hasEndOffset) searchOptions.add(SearchOptions.WANT_ENDPOS);
|
if (hasEndOffset) searchOptions.add(SearchOptions.WANT_ENDPOS);
|
||||||
|
|
||||||
// Uses RE_LAST. We know this is always set before being called
|
// Uses RE_LAST. We know this is always set before being called
|
||||||
TextRange range = SearchHelper.findPattern(editor, getLastUsedPattern(), startOffset, count, searchOptions);
|
TextRange range = injector.getSearchHelper().findPattern(new IjVimEditor(editor), getLastUsedPattern(), startOffset, count, searchOptions);
|
||||||
if (range == null) {
|
if (range == null) {
|
||||||
logger.warn("No range is found");
|
logger.warn("No range is found");
|
||||||
return -1;
|
return -1;
|
||||||
|
@@ -177,15 +177,6 @@ internal class VimMarkServiceImpl : VimMarkServiceBase(), PersistentStateCompone
|
|||||||
return createOrGetSystemMark(char, line, col, editor)
|
return createOrGetSystemMark(char, line, col, editor)
|
||||||
}
|
}
|
||||||
|
|
||||||
@Deprecated("Please use removeMark with other signature")
|
|
||||||
override fun removeMark(ch: Char, mark: Mark) {
|
|
||||||
if (ch.isGlobalMark()) {
|
|
||||||
removeGlobalMark(ch)
|
|
||||||
} else if (ch.isLocalMark()) {
|
|
||||||
getLocalMarks(mark.filepath).remove(ch)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
override fun removeGlobalMark(char: Char) {
|
override fun removeGlobalMark(char: Char) {
|
||||||
val mark = getGlobalMark(char)
|
val mark = getGlobalMark(char)
|
||||||
if (mark is IntellijMark) {
|
if (mark is IntellijMark) {
|
||||||
@@ -205,7 +196,7 @@ internal class VimMarkServiceImpl : VimMarkServiceBase(), PersistentStateCompone
|
|||||||
* @param event The change event
|
* @param event The change event
|
||||||
*/
|
*/
|
||||||
override fun beforeDocumentChange(event: DocumentEvent) {
|
override fun beforeDocumentChange(event: DocumentEvent) {
|
||||||
if (!VimPlugin.isEnabled()) return
|
if (VimPlugin.isNotEnabled()) return
|
||||||
if (logger.isDebugEnabled) logger.debug("MarkUpdater before, event = $event")
|
if (logger.isDebugEnabled) logger.debug("MarkUpdater before, event = $event")
|
||||||
if (event.oldLength == 0) return
|
if (event.oldLength == 0) return
|
||||||
val doc = event.document
|
val doc = event.document
|
||||||
@@ -221,7 +212,7 @@ internal class VimMarkServiceImpl : VimMarkServiceBase(), PersistentStateCompone
|
|||||||
* @param event The change event
|
* @param event The change event
|
||||||
*/
|
*/
|
||||||
override fun documentChanged(event: DocumentEvent) {
|
override fun documentChanged(event: DocumentEvent) {
|
||||||
if (!VimPlugin.isEnabled()) return
|
if (VimPlugin.isNotEnabled()) return
|
||||||
if (logger.isDebugEnabled) logger.debug("MarkUpdater after, event = $event")
|
if (logger.isDebugEnabled) logger.debug("MarkUpdater after, event = $event")
|
||||||
if (event.newLength == 0 || event.newLength == 1 && event.newFragment[0] != '\n') return
|
if (event.newLength == 0 || event.newLength == 1 && event.newFragment[0] != '\n') return
|
||||||
val doc = event.document
|
val doc = event.document
|
||||||
@@ -242,7 +233,7 @@ internal class VimMarkServiceImpl : VimMarkServiceBase(), PersistentStateCompone
|
|||||||
|
|
||||||
class VimBookmarksListener(private val myProject: Project) : BookmarksListener {
|
class VimBookmarksListener(private val myProject: Project) : BookmarksListener {
|
||||||
override fun bookmarkAdded(group: BookmarkGroup, bookmark: Bookmark) {
|
override fun bookmarkAdded(group: BookmarkGroup, bookmark: Bookmark) {
|
||||||
if (!VimPlugin.isEnabled()) return
|
if (VimPlugin.isNotEnabled()) return
|
||||||
if (!injector.globalIjOptions().ideamarks) {
|
if (!injector.globalIjOptions().ideamarks) {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
@@ -255,7 +246,7 @@ internal class VimMarkServiceImpl : VimMarkServiceBase(), PersistentStateCompone
|
|||||||
}
|
}
|
||||||
|
|
||||||
override fun bookmarkRemoved(group: BookmarkGroup, bookmark: Bookmark) {
|
override fun bookmarkRemoved(group: BookmarkGroup, bookmark: Bookmark) {
|
||||||
if (!VimPlugin.isEnabled()) return
|
if (VimPlugin.isNotEnabled()) return
|
||||||
if (!injector.globalIjOptions().ideamarks) {
|
if (!injector.globalIjOptions().ideamarks) {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
@@ -279,16 +270,6 @@ internal class VimMarkServiceImpl : VimMarkServiceBase(), PersistentStateCompone
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* COMPATIBILITY-LAYER: Method added
|
|
||||||
* Please see: [doc](https://jb.gg/zo8n0r)
|
|
||||||
*
|
|
||||||
*/
|
|
||||||
@Deprecated("Please use method with VimEditor")
|
|
||||||
fun saveJumpLocation(editor: Editor?) {
|
|
||||||
injector.jumpService.saveJumpLocation(IjVimEditor(editor!!))
|
|
||||||
}
|
|
||||||
|
|
||||||
companion object {
|
companion object {
|
||||||
private const val SAVE_MARK_COUNT = 20
|
private const val SAVE_MARK_COUNT = 20
|
||||||
private val logger = Logger.getInstance(
|
private val logger = Logger.getInstance(
|
||||||
|
@@ -27,15 +27,12 @@ import com.maddyhome.idea.vim.api.getLineEndOffset
|
|||||||
import com.maddyhome.idea.vim.api.globalOptions
|
import com.maddyhome.idea.vim.api.globalOptions
|
||||||
import com.maddyhome.idea.vim.api.injector
|
import com.maddyhome.idea.vim.api.injector
|
||||||
import com.maddyhome.idea.vim.api.setChangeMarks
|
import com.maddyhome.idea.vim.api.setChangeMarks
|
||||||
import com.maddyhome.idea.vim.state.mode.SelectionType
|
|
||||||
import com.maddyhome.idea.vim.state.mode.isBlock
|
|
||||||
import com.maddyhome.idea.vim.state.mode.isChar
|
|
||||||
import com.maddyhome.idea.vim.state.mode.isLine
|
|
||||||
import com.maddyhome.idea.vim.common.TextRange
|
import com.maddyhome.idea.vim.common.TextRange
|
||||||
import com.maddyhome.idea.vim.diagnostic.debug
|
import com.maddyhome.idea.vim.diagnostic.debug
|
||||||
import com.maddyhome.idea.vim.helper.EditorHelper
|
import com.maddyhome.idea.vim.helper.EditorHelper
|
||||||
import com.maddyhome.idea.vim.helper.RWLockLabel
|
import com.maddyhome.idea.vim.helper.RWLockLabel
|
||||||
import com.maddyhome.idea.vim.helper.moveToInlayAwareOffset
|
import com.maddyhome.idea.vim.helper.moveToInlayAwareOffset
|
||||||
|
import com.maddyhome.idea.vim.ide.isClionNova
|
||||||
import com.maddyhome.idea.vim.mark.VimMarkConstants.MARK_CHANGE_POS
|
import com.maddyhome.idea.vim.mark.VimMarkConstants.MARK_CHANGE_POS
|
||||||
import com.maddyhome.idea.vim.newapi.IjVimCaret
|
import com.maddyhome.idea.vim.newapi.IjVimCaret
|
||||||
import com.maddyhome.idea.vim.newapi.IjVimEditor
|
import com.maddyhome.idea.vim.newapi.IjVimEditor
|
||||||
@@ -48,6 +45,10 @@ import com.maddyhome.idea.vim.put.PutData
|
|||||||
import com.maddyhome.idea.vim.put.VimPasteProvider
|
import com.maddyhome.idea.vim.put.VimPasteProvider
|
||||||
import com.maddyhome.idea.vim.put.VimPutBase
|
import com.maddyhome.idea.vim.put.VimPutBase
|
||||||
import com.maddyhome.idea.vim.register.RegisterConstants
|
import com.maddyhome.idea.vim.register.RegisterConstants
|
||||||
|
import com.maddyhome.idea.vim.state.mode.SelectionType
|
||||||
|
import com.maddyhome.idea.vim.state.mode.isBlock
|
||||||
|
import com.maddyhome.idea.vim.state.mode.isChar
|
||||||
|
import com.maddyhome.idea.vim.state.mode.isLine
|
||||||
import java.awt.datatransfer.DataFlavor
|
import java.awt.datatransfer.DataFlavor
|
||||||
|
|
||||||
internal class PutGroup : VimPutBase() {
|
internal class PutGroup : VimPutBase() {
|
||||||
@@ -189,7 +190,7 @@ internal class PutGroup : VimPutBase() {
|
|||||||
endOffset: Int,
|
endOffset: Int,
|
||||||
): Int {
|
): Int {
|
||||||
// Temp fix for VIM-2808. Should be removed after rider will fix it's issues
|
// Temp fix for VIM-2808. Should be removed after rider will fix it's issues
|
||||||
if (PlatformUtils.isRider()) return endOffset
|
if (PlatformUtils.isRider() || isClionNova()) return endOffset
|
||||||
|
|
||||||
val startLine = editor.offsetToBufferPosition(startOffset).line
|
val startLine = editor.offsetToBufferPosition(startOffset).line
|
||||||
val endLine = editor.offsetToBufferPosition(endOffset - 1).line
|
val endLine = editor.offsetToBufferPosition(endOffset - 1).line
|
||||||
|
@@ -40,9 +40,15 @@ internal object IdeaSelectionControl {
|
|||||||
* This method should be in sync with [predictMode]
|
* This method should be in sync with [predictMode]
|
||||||
*
|
*
|
||||||
* Control unexpected (non vim) selection change and adjust a mode to it. The new mode is not enabled immediately,
|
* Control unexpected (non vim) selection change and adjust a mode to it. The new mode is not enabled immediately,
|
||||||
* but with some delay (using [VimVisualTimer])
|
* but with some delay (using [VimVisualTimer]). The delay is used because some platform functionality
|
||||||
|
* makes features by using selection. E.g. PyCharm unindent firstly select the indenting then applies delete action.
|
||||||
|
* Such "quick" selection breaks IdeaVim behaviour.
|
||||||
*
|
*
|
||||||
* See [VimVisualTimer] to more info.
|
* See [VimVisualTimer] to more info.
|
||||||
|
*
|
||||||
|
* XXX: This method can be split into "change calculation" and "change apply". In this way, we would be able
|
||||||
|
* to calculate if we need to make a change or not and reduce the number of these calls.
|
||||||
|
* If this refactoring ever is applied, please add `assertNull(VimVisualTimer.timer)` to `tearDown` of VimTestCase.
|
||||||
*/
|
*/
|
||||||
fun controlNonVimSelectionChange(
|
fun controlNonVimSelectionChange(
|
||||||
editor: Editor,
|
editor: Editor,
|
||||||
@@ -50,6 +56,7 @@ internal object IdeaSelectionControl {
|
|||||||
) {
|
) {
|
||||||
VimVisualTimer.singleTask(editor.vim.mode) { initialMode ->
|
VimVisualTimer.singleTask(editor.vim.mode) { initialMode ->
|
||||||
|
|
||||||
|
if (VimPlugin.isNotEnabled()) return@singleTask
|
||||||
if (editor.isIdeaVimDisabledHere) return@singleTask
|
if (editor.isIdeaVimDisabledHere) return@singleTask
|
||||||
|
|
||||||
logger.debug("Adjust non-vim selection. Source: $selectionSource, initialMode: $initialMode")
|
logger.debug("Adjust non-vim selection. Source: $selectionSource, initialMode: $initialMode")
|
||||||
@@ -121,8 +128,9 @@ internal object IdeaSelectionControl {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun dontChangeMode(editor: Editor): Boolean =
|
private fun dontChangeMode(editor: Editor): Boolean {
|
||||||
editor.isTemplateActive() && (editor.vim.isIdeaRefactorModeKeep || editor.vim.mode.hasVisualSelection)
|
return editor.isTemplateActive() && (editor.vim.isIdeaRefactorModeKeep || editor.vim.mode.hasVisualSelection)
|
||||||
|
}
|
||||||
|
|
||||||
private fun chooseNonSelectionMode(editor: Editor): Mode {
|
private fun chooseNonSelectionMode(editor: Editor): Mode {
|
||||||
val templateActive = editor.isTemplateActive()
|
val templateActive = editor.isTemplateActive()
|
||||||
|
@@ -9,10 +9,10 @@
|
|||||||
package com.maddyhome.idea.vim.group.visual
|
package com.maddyhome.idea.vim.group.visual
|
||||||
|
|
||||||
import com.maddyhome.idea.vim.api.injector
|
import com.maddyhome.idea.vim.api.injector
|
||||||
import com.maddyhome.idea.vim.state.mode.Mode
|
|
||||||
import com.maddyhome.idea.vim.group.visual.VimVisualTimer.mode
|
import com.maddyhome.idea.vim.group.visual.VimVisualTimer.mode
|
||||||
import com.maddyhome.idea.vim.group.visual.VimVisualTimer.singleTask
|
import com.maddyhome.idea.vim.group.visual.VimVisualTimer.singleTask
|
||||||
import com.maddyhome.idea.vim.newapi.globalIjOptions
|
import com.maddyhome.idea.vim.newapi.globalIjOptions
|
||||||
|
import com.maddyhome.idea.vim.state.mode.Mode
|
||||||
import java.awt.event.ActionEvent
|
import java.awt.event.ActionEvent
|
||||||
import javax.swing.Timer
|
import javax.swing.Timer
|
||||||
|
|
||||||
@@ -52,12 +52,13 @@ import javax.swing.Timer
|
|||||||
* editorHasSelection is false. Insert mode ([mode]) has no selection and editor also has no selection, so
|
* editorHasSelection is false. Insert mode ([mode]) has no selection and editor also has no selection, so
|
||||||
* no adjustment gets performed and IdeaVim stays in insert mode.
|
* no adjustment gets performed and IdeaVim stays in insert mode.
|
||||||
*/
|
*/
|
||||||
internal object VimVisualTimer {
|
// Do not remove until it's used in EasyMotion plugin in tests
|
||||||
|
public object VimVisualTimer {
|
||||||
|
|
||||||
var swingTimer: Timer? = null
|
public var swingTimer: Timer? = null
|
||||||
var mode: Mode? = null
|
public var mode: Mode? = null
|
||||||
|
|
||||||
inline fun singleTask(currentMode: Mode, crossinline task: (initialMode: Mode?) -> Unit) {
|
public inline fun singleTask(currentMode: Mode, crossinline task: (initialMode: Mode?) -> Unit) {
|
||||||
swingTimer?.stop()
|
swingTimer?.stop()
|
||||||
|
|
||||||
if (mode == null) mode = currentMode
|
if (mode == null) mode = currentMode
|
||||||
@@ -69,7 +70,7 @@ internal object VimVisualTimer {
|
|||||||
swingTimer = timer
|
swingTimer = timer
|
||||||
}
|
}
|
||||||
|
|
||||||
fun doNow() {
|
public fun doNow() {
|
||||||
val swingTimer1 = swingTimer
|
val swingTimer1 = swingTimer
|
||||||
if (swingTimer1 != null) {
|
if (swingTimer1 != null) {
|
||||||
swingTimer1.stop()
|
swingTimer1.stop()
|
||||||
@@ -79,7 +80,12 @@ internal object VimVisualTimer {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
inline fun timerAction(task: (initialMode: Mode?) -> Unit) {
|
public fun drop() {
|
||||||
|
swingTimer?.stop()
|
||||||
|
swingTimer = null
|
||||||
|
}
|
||||||
|
|
||||||
|
public inline fun timerAction(task: (initialMode: Mode?) -> Unit) {
|
||||||
task(mode)
|
task(mode)
|
||||||
swingTimer = null
|
swingTimer = null
|
||||||
mode = null
|
mode = null
|
||||||
|
@@ -1,102 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright 2003-2023 The IdeaVim authors
|
|
||||||
*
|
|
||||||
* Use of this source code is governed by an MIT-style
|
|
||||||
* license that can be found in the LICENSE.txt file or at
|
|
||||||
* https://opensource.org/licenses/MIT.
|
|
||||||
*/
|
|
||||||
|
|
||||||
package com.maddyhome.idea.vim.handler
|
|
||||||
|
|
||||||
import com.intellij.serviceContainer.BaseKeyedLazyInstance
|
|
||||||
import com.intellij.util.SmartList
|
|
||||||
import com.intellij.util.xmlb.annotations.Attribute
|
|
||||||
import com.maddyhome.idea.vim.command.MappingMode
|
|
||||||
import org.jetbrains.annotations.ApiStatus.ScheduledForRemoval
|
|
||||||
import javax.swing.KeyStroke
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Action holder for IdeaVim actions.
|
|
||||||
*
|
|
||||||
* [implementation] should be subclass of [EditorActionHandlerBase]
|
|
||||||
*
|
|
||||||
* [modes] ("mappingModes") defines the action modes. E.g. "NO" - action works in normal and op-pending modes.
|
|
||||||
* Warning: V - Visual and Select mode. X - Visual mode. (like vmap and xmap).
|
|
||||||
* Use "ALL" to enable action for all modes.
|
|
||||||
*
|
|
||||||
* [keys] comma-separated list of keys for the action. E.g. `gt,gT` - action gets executed on `gt` or `gT`
|
|
||||||
* Since xml doesn't allow using raw `<` character, use « and » symbols for mappings with modifiers.
|
|
||||||
* E.g. `«C-U»` - CTRL-U (<C-U> in vim notation)
|
|
||||||
* If you want to use exactly `<` character, replace it with `<`. E.g. `i<` - i<
|
|
||||||
* If you want to use comma in mapping, use `«COMMA»`
|
|
||||||
* Do not place a whitespace around the comma!
|
|
||||||
*
|
|
||||||
*
|
|
||||||
* !! IMPORTANT !!
|
|
||||||
* You may wonder why the extension points are used instead of any other approach to register actions.
|
|
||||||
* The reason is startup performance. Using the extension points you don't even have to load classes of actions.
|
|
||||||
* So, all actions are loaded on demand, including classes in classloader.
|
|
||||||
*/
|
|
||||||
@Deprecated(message = "Please use CommandOrMotion annotation")
|
|
||||||
@ScheduledForRemoval(inVersion = "2.9.0")
|
|
||||||
internal class ActionBeanClass : BaseKeyedLazyInstance<EditorActionHandlerBase>() {
|
|
||||||
@Attribute("implementation")
|
|
||||||
var implementation: String? = null
|
|
||||||
|
|
||||||
@Attribute("mappingModes")
|
|
||||||
var modes: String? = null
|
|
||||||
|
|
||||||
@Attribute("keys")
|
|
||||||
var keys: String? = null
|
|
||||||
|
|
||||||
val actionId: String get() = implementation?.let { EditorActionHandlerBase.getActionId(it) } ?: ""
|
|
||||||
|
|
||||||
fun getParsedKeys(): Set<List<KeyStroke>>? {
|
|
||||||
val myKeys = keys ?: return null
|
|
||||||
val escapedKeys = myKeys.splitByComma()
|
|
||||||
return EditorActionHandlerBase.parseKeysSet(escapedKeys)
|
|
||||||
}
|
|
||||||
|
|
||||||
override fun getImplementationClassName(): String? = implementation
|
|
||||||
|
|
||||||
fun getParsedModes(): Set<MappingMode>? {
|
|
||||||
val myModes = modes ?: return null
|
|
||||||
|
|
||||||
if ("ALL" == myModes) return MappingMode.ALL
|
|
||||||
|
|
||||||
val res = mutableListOf<MappingMode>()
|
|
||||||
for (c in myModes) {
|
|
||||||
when (c) {
|
|
||||||
'N' -> res += MappingMode.NORMAL
|
|
||||||
'X' -> res += MappingMode.VISUAL
|
|
||||||
'V' -> {
|
|
||||||
res += MappingMode.VISUAL
|
|
||||||
res += MappingMode.SELECT
|
|
||||||
}
|
|
||||||
'S' -> res += MappingMode.SELECT
|
|
||||||
'O' -> res += MappingMode.OP_PENDING
|
|
||||||
'I' -> res += MappingMode.INSERT
|
|
||||||
'C' -> res += MappingMode.CMD_LINE
|
|
||||||
else -> error("Wrong mapping mode: $c")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return res.toSet()
|
|
||||||
}
|
|
||||||
|
|
||||||
private fun String.splitByComma(): List<String> {
|
|
||||||
if (this.isEmpty()) return ArrayList()
|
|
||||||
val res = SmartList<String>()
|
|
||||||
var start = 0
|
|
||||||
var current = 0
|
|
||||||
while (current < this.length) {
|
|
||||||
if (this[current] == ',') {
|
|
||||||
res += this.substring(start, current)
|
|
||||||
current++
|
|
||||||
start = current
|
|
||||||
}
|
|
||||||
current++
|
|
||||||
}
|
|
||||||
res += this.substring(start, current)
|
|
||||||
return res
|
|
||||||
}
|
|
||||||
}
|
|
@@ -0,0 +1,120 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2003-2023 The IdeaVim authors
|
||||||
|
*
|
||||||
|
* Use of this source code is governed by an MIT-style
|
||||||
|
* license that can be found in the LICENSE.txt file or at
|
||||||
|
* https://opensource.org/licenses/MIT.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package com.maddyhome.idea.vim.handler
|
||||||
|
|
||||||
|
import com.intellij.openapi.actionSystem.KeyboardShortcut
|
||||||
|
import com.intellij.openapi.components.Service
|
||||||
|
import com.intellij.openapi.components.service
|
||||||
|
import com.intellij.openapi.diagnostic.logger
|
||||||
|
import com.intellij.openapi.keymap.Keymap
|
||||||
|
import com.intellij.openapi.keymap.KeymapManagerListener
|
||||||
|
import com.intellij.openapi.keymap.ex.KeymapManagerEx
|
||||||
|
import com.intellij.openapi.project.Project
|
||||||
|
import com.intellij.openapi.startup.ProjectActivity
|
||||||
|
import com.jetbrains.rd.util.ConcurrentHashMap
|
||||||
|
import com.maddyhome.idea.vim.VimPlugin
|
||||||
|
import com.maddyhome.idea.vim.api.injector
|
||||||
|
import com.maddyhome.idea.vim.api.key
|
||||||
|
import kotlinx.coroutines.CoroutineScope
|
||||||
|
import kotlinx.coroutines.FlowPreview
|
||||||
|
import kotlinx.coroutines.channels.BufferOverflow
|
||||||
|
import kotlinx.coroutines.flow.MutableSharedFlow
|
||||||
|
import kotlinx.coroutines.flow.collectLatest
|
||||||
|
import kotlinx.coroutines.flow.debounce
|
||||||
|
import kotlinx.coroutines.launch
|
||||||
|
|
||||||
|
|
||||||
|
// We use alarm with delay to avoid many actions in case many events are fired at the same time
|
||||||
|
internal val correctorRequester = MutableSharedFlow<Unit>(replay=1, onBufferOverflow = BufferOverflow.DROP_OLDEST)
|
||||||
|
|
||||||
|
private val LOG = logger<CopilotKeymapCorrector>()
|
||||||
|
|
||||||
|
internal class CopilotKeymapCorrector : ProjectActivity {
|
||||||
|
override suspend fun execute(project: Project) {
|
||||||
|
project.service<CopilotKeymapCorrectorService>().start()
|
||||||
|
correctorRequester.emit(Unit)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* At the moment of release 2023.3 there is a problem that starting a coroutine like this
|
||||||
|
* right in the project activity will block this project activity in tests.
|
||||||
|
* To avoid that, there is an intermediate service that will allow to avoid this issue.
|
||||||
|
*
|
||||||
|
* However, in general we should start this coroutine right in the [CopilotKeymapCorrector]
|
||||||
|
*/
|
||||||
|
@OptIn(FlowPreview::class)
|
||||||
|
@Service(Service.Level.PROJECT)
|
||||||
|
internal class CopilotKeymapCorrectorService(private val cs: CoroutineScope) {
|
||||||
|
fun start() {
|
||||||
|
cs.launch {
|
||||||
|
correctorRequester
|
||||||
|
.debounce(5_000)
|
||||||
|
.collectLatest { correctCopilotKeymap() }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
internal class IdeaVimCorrectorKeymapChangedListener : KeymapManagerListener {
|
||||||
|
override fun activeKeymapChanged(keymap: Keymap?) {
|
||||||
|
check(correctorRequester.tryEmit(Unit))
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun shortcutChanged(keymap: Keymap, actionId: String) {
|
||||||
|
check(correctorRequester.tryEmit(Unit))
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun shortcutChanged(keymap: Keymap, actionId: String, fromSettings: Boolean) {
|
||||||
|
check(correctorRequester.tryEmit(Unit))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private val copilotHideActionMap = ConcurrentHashMap<String, Unit>()
|
||||||
|
|
||||||
|
/**
|
||||||
|
* See VIM-3206
|
||||||
|
* The user expected to both copilot suggestion and the insert mode to be exited on a single esc.
|
||||||
|
* However, for the moment, the first esc hides copilot suggestion and the second one exits insert mode.
|
||||||
|
* To fix this, we remove the esc shortcut from the copilot action if the IdeaVim is active.
|
||||||
|
*
|
||||||
|
* This workaround is not the best solution, however, I don't see the better way with the current architecture of
|
||||||
|
* actions and EditorHandlers. Firstly, I wanted to suggest to copilot to migrate to EditorActionHandler as well,
|
||||||
|
* but this doesn't seem correct for me because in this case the user will lose an ability to change the shorcut for
|
||||||
|
* it. It seems like copilot has a similar problem as we do - we don't want to make a handler for "Editor enter action",
|
||||||
|
* but a handler for the esc key press. And, moreover, be able to communicate with other plugins about the ordering.
|
||||||
|
* Before this feature is implemented, hiding the copilot suggestion on esc looks like a good workaround.
|
||||||
|
*/
|
||||||
|
private fun correctCopilotKeymap() {
|
||||||
|
// This is needed to initialize the injector in case this verification is called to fast
|
||||||
|
VimPlugin.getInstance()
|
||||||
|
|
||||||
|
if (!enableOctopus) return
|
||||||
|
if (injector.enabler.isEnabled()) {
|
||||||
|
val keymap = KeymapManagerEx.getInstanceEx().activeKeymap
|
||||||
|
val res = keymap.getShortcuts("copilot.disposeInlays")
|
||||||
|
if (res.isEmpty()) return
|
||||||
|
|
||||||
|
|
||||||
|
val escapeShortcut = res.find { it.toString() == "[pressed ESCAPE]" } ?: return
|
||||||
|
keymap.removeShortcut("copilot.disposeInlays", escapeShortcut)
|
||||||
|
copilotHideActionMap[keymap.name] = Unit
|
||||||
|
LOG.info("Remove copilot escape shortcut from keymap ${keymap.name}")
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
copilotHideActionMap.forEach { (name, _) ->
|
||||||
|
val keymap = KeymapManagerEx.getInstanceEx().getKeymap(name) ?: return@forEach
|
||||||
|
val currentShortcuts = keymap.getShortcuts("copilot.disposeInlays")
|
||||||
|
if ("[pressed ESCAPE]" !in currentShortcuts.map { it.toString() }) {
|
||||||
|
keymap.addShortcut("copilot.disposeInlays", KeyboardShortcut(key("<esc>"), null))
|
||||||
|
}
|
||||||
|
LOG.info("Restore copilot escape shortcut in keymap ${keymap.name}")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@@ -8,11 +8,14 @@
|
|||||||
|
|
||||||
package com.maddyhome.idea.vim.handler
|
package com.maddyhome.idea.vim.handler
|
||||||
|
|
||||||
|
import com.intellij.openapi.actionSystem.IdeActions
|
||||||
import com.intellij.openapi.diagnostic.logger
|
import com.intellij.openapi.diagnostic.logger
|
||||||
import com.intellij.openapi.editor.actionSystem.EditorActionHandlerBean
|
import com.intellij.openapi.editor.actionSystem.EditorActionHandlerBean
|
||||||
import com.intellij.openapi.extensions.ExtensionPointName
|
import com.intellij.openapi.extensions.ExtensionPointName
|
||||||
|
import com.intellij.openapi.keymap.ex.KeymapManagerEx
|
||||||
import com.intellij.openapi.project.Project
|
import com.intellij.openapi.project.Project
|
||||||
import com.intellij.openapi.startup.StartupActivity
|
import com.intellij.openapi.startup.ProjectActivity
|
||||||
|
import com.maddyhome.idea.vim.api.key
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Logs the chain of handlers for esc and enter
|
* Logs the chain of handlers for esc and enter
|
||||||
@@ -26,11 +29,13 @@ import com.intellij.openapi.startup.StartupActivity
|
|||||||
* Strictly speaking, such access to the extension point is not allowed by the platform. But we can't do this thing
|
* Strictly speaking, such access to the extension point is not allowed by the platform. But we can't do this thing
|
||||||
* otherwise, so let's use it as long as we can.
|
* otherwise, so let's use it as long as we can.
|
||||||
*/
|
*/
|
||||||
internal class EditorHandlersChainLogger : StartupActivity {
|
internal class EditorHandlersChainLogger : ProjectActivity {
|
||||||
@Suppress("UnresolvedPluginConfigReference")
|
@Suppress("UnresolvedPluginConfigReference")
|
||||||
private val editorHandlers = ExtensionPointName<EditorActionHandlerBean>("com.intellij.editorActionHandler")
|
private val editorHandlers = ExtensionPointName<EditorActionHandlerBean>("com.intellij.editorActionHandler")
|
||||||
|
|
||||||
override fun runActivity(project: Project) {
|
override suspend fun execute(project: Project) {
|
||||||
|
if (!enableOctopus) return
|
||||||
|
|
||||||
val escHandlers = editorHandlers.extensionList
|
val escHandlers = editorHandlers.extensionList
|
||||||
.filter { it.action == "EditorEscape" }
|
.filter { it.action == "EditorEscape" }
|
||||||
.joinToString("\n") { it.implementationClass }
|
.joinToString("\n") { it.implementationClass }
|
||||||
@@ -40,6 +45,22 @@ internal class EditorHandlersChainLogger : StartupActivity {
|
|||||||
|
|
||||||
LOG.info("Esc handlers chain:\n$escHandlers")
|
LOG.info("Esc handlers chain:\n$escHandlers")
|
||||||
LOG.info("Enter handlers chain:\n$enterHandlers")
|
LOG.info("Enter handlers chain:\n$enterHandlers")
|
||||||
|
|
||||||
|
val keymapManager = KeymapManagerEx.getInstanceEx()
|
||||||
|
val keymap = keymapManager.activeKeymap
|
||||||
|
val keymapShortcutsForEsc = keymap.getShortcuts(IdeActions.ACTION_EDITOR_ESCAPE).joinToString()
|
||||||
|
val keymapShortcutsForEnter = keymap.getShortcuts(IdeActions.ACTION_EDITOR_ENTER).joinToString()
|
||||||
|
|
||||||
|
LOG.info("Active keymap (${keymap.name}) shortcuts for esc: $keymapShortcutsForEsc, Shortcuts for enter: $keymapShortcutsForEnter")
|
||||||
|
|
||||||
|
val actionsForEsc = keymap.getActionIds(key("<esc>")).joinToString("\n")
|
||||||
|
val actionsForEnter = keymap.getActionIds(key("<enter>")).joinToString("\n")
|
||||||
|
|
||||||
|
LOG.info(
|
||||||
|
"Also keymap (${keymap.name}) has " +
|
||||||
|
"the following actions assigned to esc:\n$actionsForEsc " +
|
||||||
|
"\nand following actions assigned to enter:\n$actionsForEnter"
|
||||||
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
companion object {
|
companion object {
|
||||||
|
156
src/main/java/com/maddyhome/idea/vim/handler/KeymapChecker.kt
Normal file
156
src/main/java/com/maddyhome/idea/vim/handler/KeymapChecker.kt
Normal file
@@ -0,0 +1,156 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2003-2023 The IdeaVim authors
|
||||||
|
*
|
||||||
|
* Use of this source code is governed by an MIT-style
|
||||||
|
* license that can be found in the LICENSE.txt file or at
|
||||||
|
* https://opensource.org/licenses/MIT.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package com.maddyhome.idea.vim.handler
|
||||||
|
|
||||||
|
import com.intellij.openapi.actionSystem.IdeActions
|
||||||
|
import com.intellij.openapi.actionSystem.KeyboardShortcut
|
||||||
|
import com.intellij.openapi.actionSystem.Shortcut
|
||||||
|
import com.intellij.openapi.components.Service
|
||||||
|
import com.intellij.openapi.components.service
|
||||||
|
import com.intellij.openapi.keymap.Keymap
|
||||||
|
import com.intellij.openapi.keymap.KeymapManagerListener
|
||||||
|
import com.intellij.openapi.keymap.ex.KeymapManagerEx
|
||||||
|
import com.intellij.openapi.project.Project
|
||||||
|
import com.intellij.openapi.startup.ProjectActivity
|
||||||
|
import com.maddyhome.idea.vim.VimPlugin
|
||||||
|
import com.maddyhome.idea.vim.api.injector
|
||||||
|
import com.maddyhome.idea.vim.api.key
|
||||||
|
import kotlinx.coroutines.CoroutineScope
|
||||||
|
import kotlinx.coroutines.FlowPreview
|
||||||
|
import kotlinx.coroutines.channels.BufferOverflow
|
||||||
|
import kotlinx.coroutines.flow.MutableSharedFlow
|
||||||
|
import kotlinx.coroutines.flow.collectLatest
|
||||||
|
import kotlinx.coroutines.flow.debounce
|
||||||
|
import kotlinx.coroutines.launch
|
||||||
|
import javax.swing.KeyStroke
|
||||||
|
|
||||||
|
// We use alarm with delay to avoid many notifications in case many events are fired at the same time
|
||||||
|
internal val keyCheckRequests = MutableSharedFlow<Unit>(replay=1, onBufferOverflow = BufferOverflow.DROP_OLDEST)
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This checker verifies that the keymap has a correct configuration that is required for IdeaVim plugin
|
||||||
|
*/
|
||||||
|
internal class KeymapChecker : ProjectActivity {
|
||||||
|
override suspend fun execute(project: Project) {
|
||||||
|
project.service<KeymapCheckerService>().start()
|
||||||
|
keyCheckRequests.emit(Unit)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* At the moment of release 2023.3 there is a problem that starting a coroutine like this
|
||||||
|
* right in the project activity will block this project activity in tests.
|
||||||
|
* To avoid that, there is an intermediate service that will allow to avoid this issue.
|
||||||
|
*
|
||||||
|
* However, in general we should start this coroutine right in the [KeymapChecker]
|
||||||
|
*/
|
||||||
|
@OptIn(FlowPreview::class)
|
||||||
|
@Service(Service.Level.PROJECT)
|
||||||
|
internal class KeymapCheckerService(private val cs: CoroutineScope) {
|
||||||
|
fun start() {
|
||||||
|
cs.launch {
|
||||||
|
keyCheckRequests
|
||||||
|
.debounce(5_000)
|
||||||
|
.collectLatest { verifyKeymap() }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
internal class IdeaVimKeymapChangedListener : KeymapManagerListener {
|
||||||
|
override fun activeKeymapChanged(keymap: Keymap?) {
|
||||||
|
check(keyCheckRequests.tryEmit(Unit))
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun shortcutChanged(keymap: Keymap, actionId: String) {
|
||||||
|
check(keyCheckRequests.tryEmit(Unit))
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun shortcutChanged(keymap: Keymap, actionId: String, fromSettings: Boolean) {
|
||||||
|
check(keyCheckRequests.tryEmit(Unit))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* After migration to the editor action handlers, we have to make sure that the keymap has a correct configuration.
|
||||||
|
* For example, that esc key is assigned to esc editor action
|
||||||
|
*
|
||||||
|
* Usually this is not a problem because this is a standard mapping, but the problem may appear in a misconfiguration
|
||||||
|
* like it was in VIM-3204
|
||||||
|
*/
|
||||||
|
private fun verifyKeymap() {
|
||||||
|
// This is needed to initialize the injector in case this verification is called to fast
|
||||||
|
VimPlugin.getInstance()
|
||||||
|
|
||||||
|
if (!enableOctopus) return
|
||||||
|
if (!injector.enabler.isEnabled()) return
|
||||||
|
|
||||||
|
val keymap = KeymapManagerEx.getInstanceEx().activeKeymap
|
||||||
|
val keymapShortcutsForEsc = keymap.getShortcuts(IdeActions.ACTION_EDITOR_ESCAPE)
|
||||||
|
val keymapShortcutsForEnter = keymap.getShortcuts(IdeActions.ACTION_EDITOR_ENTER)
|
||||||
|
|
||||||
|
val issues = ArrayList<KeyMapIssue>()
|
||||||
|
val correctShortcutMissing = keymapShortcutsForEsc
|
||||||
|
.filterIsInstance<KeyboardShortcut>()
|
||||||
|
.none { it.firstKeyStroke.toString() == "pressed ESCAPE" && it.secondKeyStroke == null }
|
||||||
|
|
||||||
|
// We also check if there are any shortcuts starting from esc and with a second key. This should also be removed.
|
||||||
|
// For example, VIM-3162 has a case when two escapes were assigned to editor escape action
|
||||||
|
val shortcutsStartingFromEsc = keymapShortcutsForEsc
|
||||||
|
.filterIsInstance<KeyboardShortcut>()
|
||||||
|
.filter { it.firstKeyStroke.toString() == "pressed ESCAPE" && it.secondKeyStroke != null }
|
||||||
|
if (correctShortcutMissing) {
|
||||||
|
issues += KeyMapIssue.AddShortcut(
|
||||||
|
"esc",
|
||||||
|
"editor escape",
|
||||||
|
IdeActions.ACTION_EDITOR_ESCAPE,
|
||||||
|
key("<esc>")
|
||||||
|
)
|
||||||
|
}
|
||||||
|
shortcutsStartingFromEsc.forEach {
|
||||||
|
issues += KeyMapIssue.RemoveShortcut("editor escape", IdeActions.ACTION_EDITOR_ESCAPE, it)
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
val correctEnterShortcutMissing = keymapShortcutsForEnter
|
||||||
|
.filterIsInstance<KeyboardShortcut>()
|
||||||
|
.none { it.firstKeyStroke.toString() == "pressed ENTER" && it.secondKeyStroke == null }
|
||||||
|
val shortcutsStartingFromEnter = keymapShortcutsForEnter
|
||||||
|
.filterIsInstance<KeyboardShortcut>()
|
||||||
|
.filter { it.firstKeyStroke.toString() == "pressed ENTER" && it.secondKeyStroke != null }
|
||||||
|
if (correctEnterShortcutMissing) {
|
||||||
|
issues += KeyMapIssue.AddShortcut(
|
||||||
|
"enter",
|
||||||
|
"editor enter",
|
||||||
|
IdeActions.ACTION_EDITOR_ENTER,
|
||||||
|
key("<enter>")
|
||||||
|
)
|
||||||
|
}
|
||||||
|
shortcutsStartingFromEnter.forEach {
|
||||||
|
issues += KeyMapIssue.RemoveShortcut("editor enter", IdeActions.ACTION_EDITOR_ENTER, it)
|
||||||
|
}
|
||||||
|
|
||||||
|
if (issues.isNotEmpty()) {
|
||||||
|
VimPlugin.getNotifications(null).notifyKeymapIssues(issues)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
internal sealed interface KeyMapIssue {
|
||||||
|
data class AddShortcut(
|
||||||
|
val key: String,
|
||||||
|
val action: String,
|
||||||
|
val actionId: String,
|
||||||
|
val keyStroke: KeyStroke,
|
||||||
|
) : KeyMapIssue
|
||||||
|
|
||||||
|
data class RemoveShortcut(
|
||||||
|
val action: String,
|
||||||
|
val actionId: String,
|
||||||
|
val shortcut: Shortcut,
|
||||||
|
): KeyMapIssue
|
||||||
|
}
|
@@ -10,6 +10,7 @@ package com.maddyhome.idea.vim.handler
|
|||||||
|
|
||||||
import com.intellij.codeInsight.editorActions.AutoHardWrapHandler
|
import com.intellij.codeInsight.editorActions.AutoHardWrapHandler
|
||||||
import com.intellij.codeInsight.lookup.LookupManager
|
import com.intellij.codeInsight.lookup.LookupManager
|
||||||
|
import com.intellij.formatting.LineWrappingUtil
|
||||||
import com.intellij.ide.DataManager
|
import com.intellij.ide.DataManager
|
||||||
import com.intellij.openapi.actionSystem.DataContext
|
import com.intellij.openapi.actionSystem.DataContext
|
||||||
import com.intellij.openapi.application.ApplicationManager
|
import com.intellij.openapi.application.ApplicationManager
|
||||||
@@ -18,6 +19,7 @@ import com.intellij.openapi.diagnostic.logger
|
|||||||
import com.intellij.openapi.editor.Caret
|
import com.intellij.openapi.editor.Caret
|
||||||
import com.intellij.openapi.editor.Editor
|
import com.intellij.openapi.editor.Editor
|
||||||
import com.intellij.openapi.editor.actionSystem.EditorActionHandler
|
import com.intellij.openapi.editor.actionSystem.EditorActionHandler
|
||||||
|
import com.intellij.openapi.editor.actions.SplitLineAction
|
||||||
import com.intellij.openapi.editor.impl.CaretModelImpl
|
import com.intellij.openapi.editor.impl.CaretModelImpl
|
||||||
import com.intellij.openapi.fileEditor.FileDocumentManager
|
import com.intellij.openapi.fileEditor.FileDocumentManager
|
||||||
import com.intellij.openapi.util.Key
|
import com.intellij.openapi.util.Key
|
||||||
@@ -25,6 +27,7 @@ import com.intellij.openapi.util.UserDataHolder
|
|||||||
import com.intellij.openapi.util.removeUserData
|
import com.intellij.openapi.util.removeUserData
|
||||||
import com.maddyhome.idea.vim.KeyHandler
|
import com.maddyhome.idea.vim.KeyHandler
|
||||||
import com.maddyhome.idea.vim.VimPlugin
|
import com.maddyhome.idea.vim.VimPlugin
|
||||||
|
import com.maddyhome.idea.vim.api.globalOptions
|
||||||
import com.maddyhome.idea.vim.api.injector
|
import com.maddyhome.idea.vim.api.injector
|
||||||
import com.maddyhome.idea.vim.api.key
|
import com.maddyhome.idea.vim.api.key
|
||||||
import com.maddyhome.idea.vim.group.IjOptionConstants
|
import com.maddyhome.idea.vim.group.IjOptionConstants
|
||||||
@@ -50,7 +53,7 @@ internal val commandContinuation = Key.create<EditorActionHandler>("commandConti
|
|||||||
*/
|
*/
|
||||||
internal class CaretShapeEnterEditorHandler(private val nextHandler: EditorActionHandler) : EditorActionHandler() {
|
internal class CaretShapeEnterEditorHandler(private val nextHandler: EditorActionHandler) : EditorActionHandler() {
|
||||||
override fun doExecute(editor: Editor, caret: Caret?, dataContext: DataContext?) {
|
override fun doExecute(editor: Editor, caret: Caret?, dataContext: DataContext?) {
|
||||||
if (VimPlugin.isEnabled()) {
|
if (VimPlugin.isEnabled() && enableOctopus) {
|
||||||
invokeLater {
|
invokeLater {
|
||||||
editor.updateCaretsVisualAttributes()
|
editor.updateCaretsVisualAttributes()
|
||||||
}
|
}
|
||||||
@@ -95,7 +98,15 @@ internal abstract class OctopusHandler(private val nextHandler: EditorActionHand
|
|||||||
// the condition (see VIM-3103 for example).
|
// the condition (see VIM-3103 for example).
|
||||||
// Since we can't make sure we don't execute `runForEachCaret`, we have to "escape" out of this function. This is
|
// Since we can't make sure we don't execute `runForEachCaret`, we have to "escape" out of this function. This is
|
||||||
// done by scheduling the execution of our code later via the invokeLater function.
|
// done by scheduling the execution of our code later via the invokeLater function.
|
||||||
ApplicationManager.getApplication().invokeLater(executionHandler)
|
//
|
||||||
|
// We run this job only once for a primary caret. In the handler itself, we'll multiply the execution by the
|
||||||
|
// number of carets. If we run this job for each caret, we may end up in the issue like VIM-3186.
|
||||||
|
// However, I think that we may do some refactoring to run this job for each caret (if needed).
|
||||||
|
//
|
||||||
|
// For the moment, the known case when the caret is null - work in injected editor - VIM-3195
|
||||||
|
if (caret == null || caret == editor.caretModel.primaryCaret) {
|
||||||
|
ApplicationManager.getApplication().invokeLater(executionHandler)
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
executionHandler()
|
executionHandler()
|
||||||
}
|
}
|
||||||
@@ -106,14 +117,19 @@ internal abstract class OctopusHandler(private val nextHandler: EditorActionHand
|
|||||||
|
|
||||||
private fun executeInInvokeLater(editor: Editor): Boolean {
|
private fun executeInInvokeLater(editor: Editor): Boolean {
|
||||||
// Currently we have a workaround for the PY console VIM-3157
|
// Currently we have a workaround for the PY console VIM-3157
|
||||||
if (FileDocumentManager.getInstance().getFile(editor.document)?.name == "Python Console.py") return false
|
val fileName = FileDocumentManager.getInstance().getFile(editor.document)?.name
|
||||||
|
if (
|
||||||
|
fileName == "Python Console.py" || // This is the name in 232+
|
||||||
|
fileName == "Python Console" // This is the name in 231
|
||||||
|
) return false
|
||||||
return (editor.caretModel as? CaretModelImpl)?.isIteratingOverCarets ?: true
|
return (editor.caretModel as? CaretModelImpl)?.isIteratingOverCarets ?: true
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun isThisHandlerEnabled(editor: Editor, caret: Caret?, dataContext: DataContext?): Boolean {
|
private fun isThisHandlerEnabled(editor: Editor, caret: Caret?, dataContext: DataContext?): Boolean {
|
||||||
if (!VimPlugin.isEnabled()) return false
|
if (VimPlugin.isNotEnabled()) return false
|
||||||
if (!isHandlerEnabled(editor, dataContext)) return false
|
if (!isHandlerEnabled(editor, dataContext)) return false
|
||||||
if (isNotActualKeyPress(dataContext)) return false
|
if (isNotActualKeyPress(dataContext)) return false
|
||||||
|
if (!enableOctopus) return false
|
||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -131,7 +147,20 @@ internal abstract class OctopusHandler(private val nextHandler: EditorActionHand
|
|||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
|
|
||||||
if (dataManager.loadFromDataContext(dataContext, ShiftEnterDetector.Util.key) == true) {
|
// From VIM-3177
|
||||||
|
val wrapLongLineDuringFormattingInProgress = dataManager
|
||||||
|
.loadFromDataContext(dataContext, LineWrappingUtil.WRAP_LONG_LINE_DURING_FORMATTING_IN_PROGRESS_KEY)
|
||||||
|
if (wrapLongLineDuringFormattingInProgress == true) {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
// From VIM-3203
|
||||||
|
val splitLineInProgress = dataManager.loadFromDataContext(dataContext, SplitLineAction.SPLIT_LINE_KEY)
|
||||||
|
if (splitLineInProgress == true) {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
if (dataManager.loadFromDataContext(dataContext, StartNewLineDetectorBase.Util.key) == true) {
|
||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -202,7 +231,7 @@ internal class VimEscHandler(nextHandler: EditorActionHandler) : VimKeyHandler(n
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Rider uses a separate handler for esc to close the completion. IdeaOnlyEscapeHandlerAction is especially
|
* Rider (and CLion Nova) uses a separate handler for esc to close the completion. IdeaOnlyEscapeHandlerAction is especially
|
||||||
* designer to get all the esc presses, and if there is a completion close it and do not pass the execution further.
|
* designer to get all the esc presses, and if there is a completion close it and do not pass the execution further.
|
||||||
* This doesn't work the same as in IJ.
|
* This doesn't work the same as in IJ.
|
||||||
* In IdeaVim, we'd like to exit insert mode on closing completion. This is a requirement as the change of this
|
* In IdeaVim, we'd like to exit insert mode on closing completion. This is a requirement as the change of this
|
||||||
@@ -215,6 +244,7 @@ internal class VimEscForRiderHandler(nextHandler: EditorActionHandler) : VimKeyH
|
|||||||
override val key: String = "<Esc>"
|
override val key: String = "<Esc>"
|
||||||
|
|
||||||
override fun isHandlerEnabled(editor: Editor, dataContext: DataContext?): Boolean {
|
override fun isHandlerEnabled(editor: Editor, dataContext: DataContext?): Boolean {
|
||||||
|
if (!enableOctopus) return false
|
||||||
return LookupManager.getActiveLookup(editor) != null
|
return LookupManager.getActiveLookup(editor) != null
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -230,7 +260,9 @@ internal class VimEscForRiderHandler(nextHandler: EditorActionHandler) : VimKeyH
|
|||||||
*/
|
*/
|
||||||
internal class VimEscLoggerHandler(private val nextHandler: EditorActionHandler) : EditorActionHandler() {
|
internal class VimEscLoggerHandler(private val nextHandler: EditorActionHandler) : EditorActionHandler() {
|
||||||
override fun doExecute(editor: Editor, caret: Caret?, dataContext: DataContext?) {
|
override fun doExecute(editor: Editor, caret: Caret?, dataContext: DataContext?) {
|
||||||
LOG.info("Esc pressed")
|
if (enableOctopus) {
|
||||||
|
LOG.info("Esc pressed")
|
||||||
|
}
|
||||||
nextHandler.execute(editor, caret, dataContext)
|
nextHandler.execute(editor, caret, dataContext)
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -244,13 +276,21 @@ internal class VimEscLoggerHandler(private val nextHandler: EditorActionHandler)
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Workaround to support shift-enter in normal mode.
|
* Workaround to support "Start New Line" action in normal mode.
|
||||||
* IJ executes enter handler on shift-enter. This causes an issue that IdeaVim thinks that this is just an enter key.
|
* IJ executes enter handler on "Start New Line". This causes an issue that IdeaVim thinks that this is just an enter key.
|
||||||
* This thing should be refactored, but for now we'll use this workaround VIM-3159
|
* This thing should be refactored, but for now we'll use this workaround VIM-3159
|
||||||
|
*
|
||||||
|
* The Same thing happens with "Start New Line Before Current" action.
|
||||||
*/
|
*/
|
||||||
internal class ShiftEnterDetector(private val nextHandler: EditorActionHandler) : EditorActionHandler() {
|
internal class StartNewLineDetector(nextHandler: EditorActionHandler) : StartNewLineDetectorBase(nextHandler)
|
||||||
|
internal class StartNewLineBeforeCurrentDetector(nextHandler: EditorActionHandler) :
|
||||||
|
StartNewLineDetectorBase(nextHandler)
|
||||||
|
|
||||||
|
internal open class StartNewLineDetectorBase(private val nextHandler: EditorActionHandler) : EditorActionHandler() {
|
||||||
override fun doExecute(editor: Editor, caret: Caret?, dataContext: DataContext?) {
|
override fun doExecute(editor: Editor, caret: Caret?, dataContext: DataContext?) {
|
||||||
DataManager.getInstance().saveInDataContext(dataContext, Util.key, true)
|
if (enableOctopus) {
|
||||||
|
DataManager.getInstance().saveInDataContext(dataContext, Util.key, true)
|
||||||
|
}
|
||||||
nextHandler.execute(editor, caret, dataContext)
|
nextHandler.execute(editor, caret, dataContext)
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -259,7 +299,7 @@ internal class ShiftEnterDetector(private val nextHandler: EditorActionHandler)
|
|||||||
}
|
}
|
||||||
|
|
||||||
object Util {
|
object Util {
|
||||||
val key = Key.create<Boolean>("vim.is.shift.enter")
|
val key = Key.create<Boolean>("vim.is.start.new.line")
|
||||||
}
|
}
|
||||||
|
|
||||||
companion object {
|
companion object {
|
||||||
@@ -278,7 +318,9 @@ internal class ShiftEnterDetector(private val nextHandler: EditorActionHandler)
|
|||||||
*/
|
*/
|
||||||
internal class VimEnterLoggerHandler(private val nextHandler: EditorActionHandler) : EditorActionHandler() {
|
internal class VimEnterLoggerHandler(private val nextHandler: EditorActionHandler) : EditorActionHandler() {
|
||||||
override fun doExecute(editor: Editor, caret: Caret?, dataContext: DataContext?) {
|
override fun doExecute(editor: Editor, caret: Caret?, dataContext: DataContext?) {
|
||||||
LOG.info("Enter pressed")
|
if (enableOctopus) {
|
||||||
|
LOG.info("Enter pressed")
|
||||||
|
}
|
||||||
nextHandler.execute(editor, caret, dataContext)
|
nextHandler.execute(editor, caret, dataContext)
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -308,12 +350,16 @@ internal abstract class VimKeyHandler(nextHandler: EditorActionHandler?) : Octop
|
|||||||
}
|
}
|
||||||
|
|
||||||
internal fun isOctopusEnabled(s: KeyStroke, editor: Editor): Boolean {
|
internal fun isOctopusEnabled(s: KeyStroke, editor: Editor): Boolean {
|
||||||
|
if (!enableOctopus) return false
|
||||||
// CMD line has a different processing mechanizm: the processing actions are registered
|
// CMD line has a different processing mechanizm: the processing actions are registered
|
||||||
// for the input field component. These keys are not dispatched via the octopus handler.
|
// for the input field component. These keys are not dispatched via the octopus handler.
|
||||||
if (editor.vim.mode is Mode.CMD_LINE) return false
|
if (editor.vim.mode is Mode.CMD_LINE) return false
|
||||||
when (s.keyCode) {
|
when {
|
||||||
KeyEvent.VK_ENTER -> return true
|
s.keyCode == KeyEvent.VK_ENTER && s.modifiers == 0 -> return true
|
||||||
KeyEvent.VK_ESCAPE -> return true
|
s.keyCode == KeyEvent.VK_ESCAPE && s.modifiers == 0 -> return true
|
||||||
}
|
}
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
|
|
||||||
|
internal val enableOctopus: Boolean
|
||||||
|
get() = injector.globalOptions().octopushandler
|
||||||
|
@@ -81,7 +81,7 @@ private fun Editor.guicursorMode(): GuiCursorMode {
|
|||||||
private fun isBlockCursorOverride() = EditorSettingsExternalizable.getInstance().isBlockCursor
|
private fun isBlockCursorOverride() = EditorSettingsExternalizable.getInstance().isBlockCursor
|
||||||
|
|
||||||
private fun Editor.updatePrimaryCaretVisualAttributes() {
|
private fun Editor.updatePrimaryCaretVisualAttributes() {
|
||||||
if (!VimPlugin.isEnabled()) thisLogger().error("The caret attributes should not be updated if the IdeaVim is disabled")
|
if (VimPlugin.isNotEnabled()) thisLogger().error("The caret attributes should not be updated if the IdeaVim is disabled")
|
||||||
caretModel.primaryCaret.visualAttributes = AttributesCache.getCaretVisualAttributes(this)
|
caretModel.primaryCaret.visualAttributes = AttributesCache.getCaretVisualAttributes(this)
|
||||||
|
|
||||||
// Make sure the caret is visible as soon as it's set. It might be invisible while blinking
|
// Make sure the caret is visible as soon as it's set. It might be invisible while blinking
|
||||||
@@ -89,7 +89,7 @@ private fun Editor.updatePrimaryCaretVisualAttributes() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
private fun Editor.updateSecondaryCaretsVisualAttributes() {
|
private fun Editor.updateSecondaryCaretsVisualAttributes() {
|
||||||
if (!VimPlugin.isEnabled()) thisLogger().error("The caret attributes should not be updated if the IdeaVim is disabled")
|
if (VimPlugin.isNotEnabled()) thisLogger().error("The caret attributes should not be updated if the IdeaVim is disabled")
|
||||||
// IntelliJ simulates visual block with multiple carets with selections. Do our best to hide them
|
// IntelliJ simulates visual block with multiple carets with selections. Do our best to hide them
|
||||||
val attributes = if (this.vim.inBlockSelection) HIDDEN else AttributesCache.getCaretVisualAttributes(this)
|
val attributes = if (this.vim.inBlockSelection) HIDDEN else AttributesCache.getCaretVisualAttributes(this)
|
||||||
this.caretModel.allCarets.forEach {
|
this.caretModel.allCarets.forEach {
|
||||||
|
@@ -36,7 +36,7 @@ public val Editor.mode: CommandState.Mode
|
|||||||
get() {
|
get() {
|
||||||
val mode = this.vim.vimStateMachine.mode
|
val mode = this.vim.vimStateMachine.mode
|
||||||
return when (mode) {
|
return when (mode) {
|
||||||
Mode.CMD_LINE -> CommandState.Mode.CMD_LINE
|
is Mode.CMD_LINE -> CommandState.Mode.CMD_LINE
|
||||||
Mode.INSERT -> CommandState.Mode.INSERT
|
Mode.INSERT -> CommandState.Mode.INSERT
|
||||||
is Mode.NORMAL -> CommandState.Mode.COMMAND
|
is Mode.NORMAL -> CommandState.Mode.COMMAND
|
||||||
is Mode.OP_PENDING -> CommandState.Mode.OP_PENDING
|
is Mode.OP_PENDING -> CommandState.Mode.OP_PENDING
|
||||||
|
@@ -46,12 +46,6 @@ public class EditorHelper {
|
|||||||
return editor.getScrollingModel().getVisibleAreaOnScrollingFinished();
|
return editor.getScrollingModel().getVisibleAreaOnScrollingFinished();
|
||||||
}
|
}
|
||||||
|
|
||||||
//("Use extension function with the same name on VimEditor")
|
|
||||||
@Deprecated
|
|
||||||
public static boolean isLineEmpty(final @NotNull Editor editor, final int line, final boolean allowBlanks) {
|
|
||||||
return EngineEditorHelperKt.isLineEmpty(new IjVimEditor(editor), line, allowBlanks);
|
|
||||||
}
|
|
||||||
|
|
||||||
public static boolean scrollVertically(@NotNull Editor editor, int verticalOffset) {
|
public static boolean scrollVertically(@NotNull Editor editor, int verticalOffset) {
|
||||||
final ScrollingModel scrollingModel = editor.getScrollingModel();
|
final ScrollingModel scrollingModel = editor.getScrollingModel();
|
||||||
final Rectangle area = scrollingModel.getVisibleAreaOnScrollingFinished();
|
final Rectangle area = scrollingModel.getVisibleAreaOnScrollingFinished();
|
||||||
|
@@ -23,43 +23,6 @@ import com.maddyhome.idea.vim.newapi.vim
|
|||||||
import com.maddyhome.idea.vim.state.mode.inBlockSelection
|
import com.maddyhome.idea.vim.state.mode.inBlockSelection
|
||||||
import java.util.stream.Collectors
|
import java.util.stream.Collectors
|
||||||
|
|
||||||
/**
|
|
||||||
* This annotation is created for test functions (methods).
|
|
||||||
* It means that the original vim behavior has small differences from behavior of IdeaVim.
|
|
||||||
* [shouldBeFixed] flag indicates whether the given functionality should be fixed
|
|
||||||
* or the given behavior is normal for IdeaVim and should be leaved as is.
|
|
||||||
*
|
|
||||||
* E.g. after execution of some commands original vim has the following text:
|
|
||||||
* Hello1
|
|
||||||
* Hello2
|
|
||||||
* Hello3
|
|
||||||
*
|
|
||||||
* But IdeaVim gives you:
|
|
||||||
* Hello1
|
|
||||||
*
|
|
||||||
* Hello2
|
|
||||||
* Hello3
|
|
||||||
*
|
|
||||||
* In this case you should still create the test function and mark this function with [VimBehaviorDiffers] annotation.
|
|
||||||
*
|
|
||||||
* Why does this annotation exist?
|
|
||||||
* After creating some functionality you can understand that IdeaVim has a bit different behavior, but you
|
|
||||||
* cannot fix it right now because of any reason (bugs in IDE,
|
|
||||||
* the impossibility of this functionality in IDEA (*[shouldBeFixed] == false*), leak of time for fixing).
|
|
||||||
* In that case, you should NOT remove the corresponding test or leave it without any marks that this test
|
|
||||||
* not fully convenient with vim, but leave the test with IdeaVim's behavior and put this annotation
|
|
||||||
* with description of how original vim works.
|
|
||||||
*
|
|
||||||
* Note that using this annotation should be avoided as much as possible and behavior of IdeaVim should be as close
|
|
||||||
* to vim as possible.
|
|
||||||
*/
|
|
||||||
@Target(AnnotationTarget.FUNCTION)
|
|
||||||
internal annotation class VimBehaviorDiffers(
|
|
||||||
val originalVimAfter: String = "",
|
|
||||||
val description: String = "",
|
|
||||||
val shouldBeFixed: Boolean = true,
|
|
||||||
)
|
|
||||||
|
|
||||||
internal fun <T : Comparable<T>> sort(a: T, b: T) = if (a > b) b to a else a to b
|
internal fun <T : Comparable<T>> sort(a: T, b: T) = if (a > b) b to a else a to b
|
||||||
|
|
||||||
// TODO Should be replaced with VimEditor.carets()
|
// TODO Should be replaced with VimEditor.carets()
|
||||||
@@ -110,7 +73,7 @@ internal fun Editor.isTemplateActive(): Boolean {
|
|||||||
}
|
}
|
||||||
|
|
||||||
private fun vimEnabled(editor: Editor?): Boolean {
|
private fun vimEnabled(editor: Editor?): Boolean {
|
||||||
if (!VimPlugin.isEnabled()) return false
|
if (VimPlugin.isNotEnabled()) return false
|
||||||
if (editor != null && editor.isIdeaVimDisabledHere) return false
|
if (editor != null && editor.isIdeaVimDisabledHere) return false
|
||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
|
@@ -15,10 +15,12 @@ import com.intellij.openapi.actionSystem.AnAction
|
|||||||
import com.intellij.openapi.actionSystem.AnActionEvent
|
import com.intellij.openapi.actionSystem.AnActionEvent
|
||||||
import com.intellij.openapi.actionSystem.AnActionResult
|
import com.intellij.openapi.actionSystem.AnActionResult
|
||||||
import com.intellij.openapi.actionSystem.DataContextWrapper
|
import com.intellij.openapi.actionSystem.DataContextWrapper
|
||||||
|
import com.intellij.openapi.actionSystem.EmptyAction
|
||||||
import com.intellij.openapi.actionSystem.IdeActions
|
import com.intellij.openapi.actionSystem.IdeActions
|
||||||
import com.intellij.openapi.actionSystem.PlatformDataKeys
|
import com.intellij.openapi.actionSystem.PlatformDataKeys
|
||||||
import com.intellij.openapi.actionSystem.ex.ActionManagerEx
|
import com.intellij.openapi.actionSystem.ex.ActionManagerEx
|
||||||
import com.intellij.openapi.actionSystem.ex.ActionUtil
|
import com.intellij.openapi.actionSystem.ex.ActionUtil
|
||||||
|
import com.intellij.openapi.actionSystem.impl.ProxyShortcutSet
|
||||||
import com.intellij.openapi.command.CommandProcessor
|
import com.intellij.openapi.command.CommandProcessor
|
||||||
import com.intellij.openapi.command.UndoConfirmationPolicy
|
import com.intellij.openapi.command.UndoConfirmationPolicy
|
||||||
import com.intellij.openapi.components.Service
|
import com.intellij.openapi.components.Service
|
||||||
@@ -39,6 +41,8 @@ import com.maddyhome.idea.vim.newapi.IjNativeAction
|
|||||||
import com.maddyhome.idea.vim.newapi.ij
|
import com.maddyhome.idea.vim.newapi.ij
|
||||||
import com.maddyhome.idea.vim.newapi.runFromVimKey
|
import com.maddyhome.idea.vim.newapi.runFromVimKey
|
||||||
import org.jetbrains.annotations.NonNls
|
import org.jetbrains.annotations.NonNls
|
||||||
|
import java.awt.Component
|
||||||
|
import javax.swing.JComponent
|
||||||
import javax.swing.SwingUtilities
|
import javax.swing.SwingUtilities
|
||||||
|
|
||||||
@Service
|
@Service
|
||||||
@@ -139,7 +143,7 @@ internal class IjActionExecutor : VimActionExecutor {
|
|||||||
manager.fireAfterActionPerformed(action, event, result!!)
|
manager.fireAfterActionPerformed(action, event, result!!)
|
||||||
}
|
}
|
||||||
if (indexError != null) {
|
if (indexError != null) {
|
||||||
ActionUtil.showDumbModeWarning(project, event)
|
ActionUtil.showDumbModeWarning(project, action, event)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -150,11 +154,44 @@ internal class IjActionExecutor : VimActionExecutor {
|
|||||||
* @param context The context to run it in
|
* @param context The context to run it in
|
||||||
*/
|
*/
|
||||||
override fun executeAction(name: @NonNls String, context: ExecutionContext): Boolean {
|
override fun executeAction(name: @NonNls String, context: ExecutionContext): Boolean {
|
||||||
val aMgr = ActionManager.getInstance()
|
val action = getAction(name, context)
|
||||||
val action = aMgr.getAction(name)
|
|
||||||
return action != null && executeAction(null, IjNativeAction(action), context)
|
return action != null && executeAction(null, IjNativeAction(action), context)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private fun getAction(name: String, context: ExecutionContext): AnAction? {
|
||||||
|
val actionManager = ActionManager.getInstance()
|
||||||
|
val action = actionManager.getAction(name)
|
||||||
|
if (action !is EmptyAction) return action
|
||||||
|
|
||||||
|
// But if the action is an instance of EmptyAction, the fun begins
|
||||||
|
var component: Component? = context.ij.getData(PlatformDataKeys.CONTEXT_COMPONENT) ?: return null
|
||||||
|
while (component != null) {
|
||||||
|
if (component !is JComponent) {
|
||||||
|
component = component.parent
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
val listOfActions = ActionUtil.getActions(component)
|
||||||
|
if (listOfActions.isEmpty()) {
|
||||||
|
component = component.getParent()
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
fun AnAction.getId(): String? {
|
||||||
|
return actionManager.getId(this)
|
||||||
|
?: (shortcutSet as? ProxyShortcutSet)?.actionId
|
||||||
|
}
|
||||||
|
|
||||||
|
for (action in listOfActions) {
|
||||||
|
if (action.getId() == name) {
|
||||||
|
return action
|
||||||
|
}
|
||||||
|
}
|
||||||
|
component = component.getParent()
|
||||||
|
}
|
||||||
|
return null
|
||||||
|
}
|
||||||
|
|
||||||
override fun executeCommand(
|
override fun executeCommand(
|
||||||
editor: VimEditor?,
|
editor: VimEditor?,
|
||||||
runnable: Runnable,
|
runnable: Runnable,
|
||||||
|
@@ -23,6 +23,8 @@ import com.intellij.psi.util.PsiTreeUtil;
|
|||||||
import com.maddyhome.idea.vim.VimPlugin;
|
import com.maddyhome.idea.vim.VimPlugin;
|
||||||
import com.maddyhome.idea.vim.api.EngineEditorHelperKt;
|
import com.maddyhome.idea.vim.api.EngineEditorHelperKt;
|
||||||
import com.maddyhome.idea.vim.api.VimEditor;
|
import com.maddyhome.idea.vim.api.VimEditor;
|
||||||
|
import com.maddyhome.idea.vim.regexp.*;
|
||||||
|
import com.maddyhome.idea.vim.regexp.match.VimMatchResult;
|
||||||
import com.maddyhome.idea.vim.state.mode.Mode;
|
import com.maddyhome.idea.vim.state.mode.Mode;
|
||||||
import com.maddyhome.idea.vim.state.VimStateMachine;
|
import com.maddyhome.idea.vim.state.VimStateMachine;
|
||||||
import com.maddyhome.idea.vim.common.CharacterPosition;
|
import com.maddyhome.idea.vim.common.CharacterPosition;
|
||||||
@@ -30,8 +32,6 @@ import com.maddyhome.idea.vim.common.Direction;
|
|||||||
import com.maddyhome.idea.vim.common.TextRange;
|
import com.maddyhome.idea.vim.common.TextRange;
|
||||||
import com.maddyhome.idea.vim.newapi.IjVimCaret;
|
import com.maddyhome.idea.vim.newapi.IjVimCaret;
|
||||||
import com.maddyhome.idea.vim.newapi.IjVimEditor;
|
import com.maddyhome.idea.vim.newapi.IjVimEditor;
|
||||||
import com.maddyhome.idea.vim.regexp.CharPointer;
|
|
||||||
import com.maddyhome.idea.vim.regexp.RegExp;
|
|
||||||
import kotlin.Pair;
|
import kotlin.Pair;
|
||||||
import org.jetbrains.annotations.Contract;
|
import org.jetbrains.annotations.Contract;
|
||||||
import org.jetbrains.annotations.NotNull;
|
import org.jetbrains.annotations.NotNull;
|
||||||
@@ -42,10 +42,10 @@ import java.util.function.Function;
|
|||||||
import java.util.regex.Matcher;
|
import java.util.regex.Matcher;
|
||||||
import java.util.regex.Pattern;
|
import java.util.regex.Pattern;
|
||||||
|
|
||||||
import static com.maddyhome.idea.vim.api.VimInjectorKt.injector;
|
import static com.maddyhome.idea.vim.api.VimInjectorKt.*;
|
||||||
import static com.maddyhome.idea.vim.api.VimInjectorKt.options;
|
|
||||||
import static com.maddyhome.idea.vim.helper.SearchHelperKtKt.checkInString;
|
import static com.maddyhome.idea.vim.helper.SearchHelperKtKt.checkInString;
|
||||||
import static com.maddyhome.idea.vim.helper.SearchHelperKtKt.shouldIgnoreCase;
|
import static com.maddyhome.idea.vim.helper.SearchHelperKtKt.shouldIgnoreCase;
|
||||||
|
import static com.maddyhome.idea.vim.newapi.IjVimInjectorKt.globalIjOptions;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Helper methods for searching text
|
* Helper methods for searching text
|
||||||
@@ -59,6 +59,8 @@ public class SearchHelper {
|
|||||||
/**
|
/**
|
||||||
* Find text matching the given pattern.
|
* Find text matching the given pattern.
|
||||||
*
|
*
|
||||||
|
* @deprecated Use IjVimSearchHelper.findPattern instead
|
||||||
|
*
|
||||||
* <p>See search.c:searchit</p>
|
* <p>See search.c:searchit</p>
|
||||||
*
|
*
|
||||||
* @param editor The editor to search in
|
* @param editor The editor to search in
|
||||||
@@ -69,12 +71,13 @@ public class SearchHelper {
|
|||||||
* @return A TextRange representing the result, or null
|
* @return A TextRange representing the result, or null
|
||||||
*/
|
*/
|
||||||
@Nullable
|
@Nullable
|
||||||
|
@Deprecated
|
||||||
public static TextRange findPattern(@NotNull Editor editor,
|
public static TextRange findPattern(@NotNull Editor editor,
|
||||||
@Nullable String pattern,
|
@Nullable String pattern,
|
||||||
int startOffset,
|
int startOffset,
|
||||||
int count,
|
int count,
|
||||||
EnumSet<SearchOptions> searchOptions) {
|
EnumSet<SearchOptions> searchOptions) {
|
||||||
if (pattern == null || pattern.length() == 0) {
|
if (pattern == null || pattern.isEmpty()) {
|
||||||
logger.warn("Pattern is null or empty. Cannot perform search");
|
logger.warn("Pattern is null or empty. Cannot perform search");
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
@@ -347,6 +350,8 @@ public class SearchHelper {
|
|||||||
/**
|
/**
|
||||||
* Find all occurrences of the pattern.
|
* Find all occurrences of the pattern.
|
||||||
*
|
*
|
||||||
|
* @deprecated Use IjVimSearchHelper.findall instead
|
||||||
|
*
|
||||||
* @param editor The editor to search in
|
* @param editor The editor to search in
|
||||||
* @param pattern The pattern to search for
|
* @param pattern The pattern to search for
|
||||||
* @param startLine The start line of the range to search for
|
* @param startLine The start line of the range to search for
|
||||||
@@ -354,12 +359,33 @@ public class SearchHelper {
|
|||||||
* @param ignoreCase Case sensitive or insensitive searching
|
* @param ignoreCase Case sensitive or insensitive searching
|
||||||
* @return A list of TextRange objects representing the results
|
* @return A list of TextRange objects representing the results
|
||||||
*/
|
*/
|
||||||
|
@Deprecated
|
||||||
public static @NotNull List<TextRange> findAll(@NotNull Editor editor,
|
public static @NotNull List<TextRange> findAll(@NotNull Editor editor,
|
||||||
@NotNull String pattern,
|
@NotNull String pattern,
|
||||||
int startLine,
|
int startLine,
|
||||||
int endLine,
|
int endLine,
|
||||||
boolean ignoreCase) {
|
boolean ignoreCase) {
|
||||||
final List<TextRange> results = Lists.newArrayList();
|
final List<TextRange> results = Lists.newArrayList();
|
||||||
|
|
||||||
|
if (globalIjOptions(injector).getUseNewRegex()) {
|
||||||
|
final List<VimRegexOptions> options = new ArrayList<>();
|
||||||
|
if (globalOptions(injector).getSmartcase()) options.add(VimRegexOptions.SMART_CASE);
|
||||||
|
if (globalOptions(injector).getIgnorecase()) options.add(VimRegexOptions.IGNORE_CASE);
|
||||||
|
VimEditor vimEditor = new IjVimEditor(editor);
|
||||||
|
try {
|
||||||
|
// TODO: we shouldn't care about the ignoreCase argument, and instead just look into the editor options.
|
||||||
|
// It would require a refactor, so for now prepend \c or \C to "force" ignoreCase
|
||||||
|
String newPattern = (ignoreCase ? "\\c" : "\\C") + pattern;
|
||||||
|
VimRegex regex = new VimRegex(newPattern);
|
||||||
|
List<VimMatchResult.Success> foundMatches = regex.findAll(vimEditor, vimEditor.getLineStartOffset(startLine), vimEditor.getLineEndOffset(endLine == -1 ? vimEditor.lineCount() - 1 : endLine) + 1, options);
|
||||||
|
for (VimMatchResult.Success match : foundMatches) results.add(match.getRange());
|
||||||
|
return results;
|
||||||
|
} catch (VimRegexException e) {
|
||||||
|
injector.getMessages().showStatusBarMessage(vimEditor, e.getMessage());
|
||||||
|
return results;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
final int lineCount = new IjVimEditor(editor).lineCount();
|
final int lineCount = new IjVimEditor(editor).lineCount();
|
||||||
final int actualEndLine = endLine == -1 ? lineCount - 1 : endLine;
|
final int actualEndLine = endLine == -1 ? lineCount - 1 : endLine;
|
||||||
|
|
||||||
@@ -402,6 +428,10 @@ public class SearchHelper {
|
|||||||
return results;
|
return results;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @deprecated Use IjVimSearchHelper.findSection instead
|
||||||
|
*/
|
||||||
|
@Deprecated
|
||||||
public static int findSection(@NotNull Editor editor, @NotNull Caret caret, char type, int dir, int count) {
|
public static int findSection(@NotNull Editor editor, @NotNull Caret caret, char type, int dir, int count) {
|
||||||
CharSequence chars = editor.getDocument().getCharsSequence();
|
CharSequence chars = editor.getDocument().getCharsSequence();
|
||||||
int line = caret.getLogicalPosition().line + dir;
|
int line = caret.getLogicalPosition().line + dir;
|
||||||
@@ -428,6 +458,10 @@ public class SearchHelper {
|
|||||||
return res;
|
return res;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @deprecated Use IjVimSearchHelper.findUnmatchedBlock instead
|
||||||
|
*/
|
||||||
|
@Deprecated
|
||||||
public static int findUnmatchedBlock(@NotNull Editor editor, @NotNull Caret caret, char type, int count) {
|
public static int findUnmatchedBlock(@NotNull Editor editor, @NotNull Caret caret, char type, int count) {
|
||||||
CharSequence chars = editor.getDocument().getCharsSequence();
|
CharSequence chars = editor.getDocument().getCharsSequence();
|
||||||
int pos = caret.getOffset();
|
int pos = caret.getOffset();
|
||||||
@@ -447,6 +481,8 @@ public class SearchHelper {
|
|||||||
/**
|
/**
|
||||||
* Find block enclosing the caret
|
* Find block enclosing the caret
|
||||||
*
|
*
|
||||||
|
* @deprecated Use IjVimSearchHelper.findBlockRange instead
|
||||||
|
*
|
||||||
* @param editor The editor to search in
|
* @param editor The editor to search in
|
||||||
* @param caret The caret currently at
|
* @param caret The caret currently at
|
||||||
* @param type The type of block, e.g. (, [, {, <
|
* @param type The type of block, e.g. (, [, {, <
|
||||||
@@ -798,6 +834,10 @@ public class SearchHelper {
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @deprecated Use IjVimSearchHelper.findBlockTagRange instead
|
||||||
|
*/
|
||||||
|
@Deprecated
|
||||||
public static @Nullable TextRange findBlockTagRange(@NotNull Editor editor,
|
public static @Nullable TextRange findBlockTagRange(@NotNull Editor editor,
|
||||||
@NotNull Caret caret,
|
@NotNull Caret caret,
|
||||||
int count,
|
int count,
|
||||||
@@ -1330,6 +1370,10 @@ public class SearchHelper {
|
|||||||
return new TextRange(start, end);
|
return new TextRange(start, end);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @deprecated Use IjVimSearchHelper.findWordUnderCursor instead
|
||||||
|
*/
|
||||||
|
@Deprecated
|
||||||
@Contract("_, _, _, _, _, _, _ -> new")
|
@Contract("_, _, _, _, _, _, _ -> new")
|
||||||
public static @NotNull TextRange findWordUnderCursor(@NotNull Editor editor,
|
public static @NotNull TextRange findWordUnderCursor(@NotNull Editor editor,
|
||||||
@NotNull Caret caret,
|
@NotNull Caret caret,
|
||||||
@@ -1515,10 +1559,16 @@ public class SearchHelper {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @deprecated Use IjVimSearchHelper.findMethodStart instead
|
||||||
|
*/
|
||||||
public static int findMethodStart(@NotNull Editor editor, @NotNull Caret caret, int count) {
|
public static int findMethodStart(@NotNull Editor editor, @NotNull Caret caret, int count) {
|
||||||
return PsiHelper.findMethodStart(editor, caret.getOffset(), count);
|
return PsiHelper.findMethodStart(editor, caret.getOffset(), count);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @deprecated Use IjVimSearchHelper.findMethodEnd instead
|
||||||
|
*/
|
||||||
public static int findMethodEnd(@NotNull Editor editor, @NotNull Caret caret, int count) {
|
public static int findMethodEnd(@NotNull Editor editor, @NotNull Caret caret, int count) {
|
||||||
return PsiHelper.findMethodEnd(editor, caret.getOffset(), count);
|
return PsiHelper.findMethodEnd(editor, caret.getOffset(), count);
|
||||||
}
|
}
|
||||||
|
@@ -26,6 +26,7 @@ import com.maddyhome.idea.vim.api.injector
|
|||||||
import com.maddyhome.idea.vim.api.options
|
import com.maddyhome.idea.vim.api.options
|
||||||
import com.maddyhome.idea.vim.common.TextRange
|
import com.maddyhome.idea.vim.common.TextRange
|
||||||
import com.maddyhome.idea.vim.ex.ranges.LineRange
|
import com.maddyhome.idea.vim.ex.ranges.LineRange
|
||||||
|
import com.maddyhome.idea.vim.newapi.IjVimEditor
|
||||||
import com.maddyhome.idea.vim.newapi.vim
|
import com.maddyhome.idea.vim.newapi.vim
|
||||||
import org.jetbrains.annotations.Contract
|
import org.jetbrains.annotations.Contract
|
||||||
import java.awt.Color
|
import java.awt.Color
|
||||||
@@ -105,7 +106,7 @@ private fun updateSearchHighlights(
|
|||||||
val startLine = searchRange?.startLine ?: 0
|
val startLine = searchRange?.startLine ?: 0
|
||||||
val endLine = searchRange?.endLine ?: -1
|
val endLine = searchRange?.endLine ?: -1
|
||||||
val results =
|
val results =
|
||||||
SearchHelper.findAll(editor, pattern, startLine, endLine, shouldIgnoreCase(pattern, shouldIgnoreSmartCase))
|
injector.searchHelper.findAll(IjVimEditor(editor), pattern, startLine, endLine, shouldIgnoreCase(pattern, shouldIgnoreSmartCase))
|
||||||
if (results.isNotEmpty()) {
|
if (results.isNotEmpty()) {
|
||||||
currentMatchOffset = findClosestMatch(editor, results, initialOffset, forwards)
|
currentMatchOffset = findClosestMatch(editor, results, initialOffset, forwards)
|
||||||
highlightSearchResults(editor, pattern, results, currentMatchOffset)
|
highlightSearchResults(editor, pattern, results, currentMatchOffset)
|
||||||
@@ -119,7 +120,7 @@ private fun updateSearchHighlights(
|
|||||||
}
|
}
|
||||||
if (shouldIgnoreSmartCase) searchOptions.add(SearchOptions.IGNORE_SMARTCASE)
|
if (shouldIgnoreSmartCase) searchOptions.add(SearchOptions.IGNORE_SMARTCASE)
|
||||||
if (!forwards) searchOptions.add(SearchOptions.BACKWARDS)
|
if (!forwards) searchOptions.add(SearchOptions.BACKWARDS)
|
||||||
val result = SearchHelper.findPattern(editor, pattern, initialOffset, 1, searchOptions)
|
val result = injector.searchHelper.findPattern(IjVimEditor(editor), pattern, initialOffset, 1, searchOptions)
|
||||||
if (result != null) {
|
if (result != null) {
|
||||||
currentMatchOffset = result.startOffset
|
currentMatchOffset = result.startOffset
|
||||||
val results = listOf(result)
|
val results = listOf(result)
|
||||||
|
@@ -31,10 +31,4 @@ public object StringHelper {
|
|||||||
return Arrays.stream(string).flatMap { o: String -> injector.parser.parseKeys(o).stream() }
|
return Arrays.stream(string).flatMap { o: String -> injector.parser.parseKeys(o).stream() }
|
||||||
.collect(Collectors.toList())
|
.collect(Collectors.toList())
|
||||||
}
|
}
|
||||||
|
|
||||||
@JvmStatic
|
|
||||||
@Deprecated("Use KeyStroke.isCloseKeyStroke", ReplaceWith("stroke.isCloseKeyStroke()"))
|
|
||||||
public fun isCloseKeyStroke(stroke: KeyStroke): Boolean {
|
|
||||||
return stroke.isCloseKeyStroke()
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
@@ -11,14 +11,15 @@ import com.google.common.collect.Lists
|
|||||||
import com.intellij.openapi.editor.Editor
|
import com.intellij.openapi.editor.Editor
|
||||||
import javax.swing.KeyStroke
|
import javax.swing.KeyStroke
|
||||||
|
|
||||||
internal class TestInputModel private constructor() {
|
// Do not remove until it's used in EasyMotion plugin in tests
|
||||||
|
public class TestInputModel private constructor() {
|
||||||
private val myKeyStrokes: MutableList<KeyStroke> = Lists.newArrayList()
|
private val myKeyStrokes: MutableList<KeyStroke> = Lists.newArrayList()
|
||||||
fun setKeyStrokes(keyStrokes: List<KeyStroke>) {
|
public fun setKeyStrokes(keyStrokes: List<KeyStroke>) {
|
||||||
myKeyStrokes.clear()
|
myKeyStrokes.clear()
|
||||||
myKeyStrokes.addAll(keyStrokes)
|
myKeyStrokes.addAll(keyStrokes)
|
||||||
}
|
}
|
||||||
|
|
||||||
fun nextKeyStroke(): KeyStroke? {
|
public fun nextKeyStroke(): KeyStroke? {
|
||||||
// Return key from the unfinished mapping
|
// Return key from the unfinished mapping
|
||||||
/*
|
/*
|
||||||
MappingStack mappingStack = KeyHandler.getInstance().getMappingStack();
|
MappingStack mappingStack = KeyHandler.getInstance().getMappingStack();
|
||||||
@@ -33,9 +34,9 @@ if (mappingStack.hasStroke()) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
companion object {
|
public companion object {
|
||||||
@JvmStatic
|
@JvmStatic
|
||||||
fun getInstance(editor: Editor): TestInputModel {
|
public fun getInstance(editor: Editor): TestInputModel {
|
||||||
var model = editor.vimTestInputModel
|
var model = editor.vimTestInputModel
|
||||||
if (model == null) {
|
if (model == null) {
|
||||||
model = TestInputModel()
|
model = TestInputModel()
|
||||||
|
@@ -17,6 +17,7 @@ import com.intellij.openapi.fileEditor.impl.text.TextEditorProvider
|
|||||||
import com.maddyhome.idea.vim.api.ExecutionContext
|
import com.maddyhome.idea.vim.api.ExecutionContext
|
||||||
import com.maddyhome.idea.vim.api.VimEditor
|
import com.maddyhome.idea.vim.api.VimEditor
|
||||||
import com.maddyhome.idea.vim.api.injector
|
import com.maddyhome.idea.vim.api.injector
|
||||||
|
import com.maddyhome.idea.vim.common.ChangesListener
|
||||||
import com.maddyhome.idea.vim.listener.SelectionVimListenerSuppressor
|
import com.maddyhome.idea.vim.listener.SelectionVimListenerSuppressor
|
||||||
import com.maddyhome.idea.vim.newapi.globalIjOptions
|
import com.maddyhome.idea.vim.newapi.globalIjOptions
|
||||||
import com.maddyhome.idea.vim.newapi.ij
|
import com.maddyhome.idea.vim.newapi.ij
|
||||||
@@ -40,22 +41,17 @@ internal class UndoRedoHelper : UndoRedoBase() {
|
|||||||
SelectionVimListenerSuppressor.lock().use { undoManager.undo(fileEditor) }
|
SelectionVimListenerSuppressor.lock().use { undoManager.undo(fileEditor) }
|
||||||
} else {
|
} else {
|
||||||
// TODO refactor me after VIM-308 when restoring selection and caret movement will be ignored by undo
|
// TODO refactor me after VIM-308 when restoring selection and caret movement will be ignored by undo
|
||||||
undoManager.undo(fileEditor)
|
editor.runWithChangeTracking {
|
||||||
if (hasSelection(editor) && undoManager.isUndoAvailable(fileEditor)) {
|
undoManager.undo(fileEditor)
|
||||||
undoManager.undo(fileEditor) // execute one more time if the previous undo just restored selection
|
|
||||||
|
// We execute undo one more time if the previous one just restored selection
|
||||||
|
if (!hasChanges && hasSelection(editor) && undoManager.isUndoAvailable(fileEditor)) {
|
||||||
|
undoManager.undo(fileEditor)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// remove selection
|
CommandProcessor.getInstance().runUndoTransparentAction {
|
||||||
editor.carets().forEach {
|
removeSelections(editor)
|
||||||
val ijCaret = it.ij
|
|
||||||
val hasSelection = ijCaret.hasSelection()
|
|
||||||
if (hasSelection) {
|
|
||||||
val selectionStart = ijCaret.selectionStart
|
|
||||||
CommandProcessor.getInstance().runUndoTransparentAction {
|
|
||||||
it.ij.removeSelection()
|
|
||||||
it.ij.moveToOffset(selectionStart)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -83,9 +79,56 @@ internal class UndoRedoHelper : UndoRedoBase() {
|
|||||||
CommandProcessor.getInstance().runUndoTransparentAction {
|
CommandProcessor.getInstance().runUndoTransparentAction {
|
||||||
editor.carets().forEach { it.ij.removeSelection() }
|
editor.carets().forEach { it.ij.removeSelection() }
|
||||||
}
|
}
|
||||||
|
// TODO refactor me after VIM-308 when restoring selection and caret movement will be ignored by undo
|
||||||
|
editor.runWithChangeTracking {
|
||||||
|
undoManager.redo(fileEditor)
|
||||||
|
|
||||||
|
// We execute undo one more time if the previous one just restored selection
|
||||||
|
if (!hasChanges && hasSelection(editor) && undoManager.isRedoAvailable(fileEditor)) {
|
||||||
|
undoManager.redo(fileEditor)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
CommandProcessor.getInstance().runUndoTransparentAction {
|
||||||
|
removeSelections(editor)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private fun removeSelections(editor: VimEditor) {
|
||||||
|
editor.carets().forEach {
|
||||||
|
val ijCaret = it.ij
|
||||||
|
if (!ijCaret.hasSelection()) return@forEach
|
||||||
|
|
||||||
|
val selectionStart = ijCaret.selectionStart
|
||||||
|
ijCaret.removeSelection()
|
||||||
|
ijCaret.moveToOffset(selectionStart)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun VimEditor.runWithChangeTracking(block: ChangeTracker.() -> Unit) {
|
||||||
|
val tracker = ChangeTracker(this)
|
||||||
|
tracker.block()
|
||||||
|
}
|
||||||
|
|
||||||
|
private class ChangeTracker(private val editor: VimEditor) {
|
||||||
|
private val initialPath = editor.getPath()
|
||||||
|
private val changeListener = object : ChangesListener {
|
||||||
|
var hasChanged = false
|
||||||
|
|
||||||
|
override fun documentChanged(change: ChangesListener.Change) {
|
||||||
|
hasChanged = true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
init {
|
||||||
|
editor.document.addChangeListener(changeListener)
|
||||||
|
}
|
||||||
|
|
||||||
|
val hasChanges: Boolean
|
||||||
|
get() = changeListener.hasChanged || initialPath != editor.getPath()
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@@ -124,10 +124,6 @@ internal var Editor.vimMorePanel: ExOutputPanel? by userData()
|
|||||||
internal var Editor.vimExOutput: ExOutputModel? by userData()
|
internal var Editor.vimExOutput: ExOutputModel? by userData()
|
||||||
internal var Editor.vimTestInputModel: TestInputModel? by userData()
|
internal var Editor.vimTestInputModel: TestInputModel? by userData()
|
||||||
|
|
||||||
/**
|
|
||||||
* Checks whether a keeping visual mode visual operator action is performed on editor.
|
|
||||||
*/
|
|
||||||
internal var Editor.vimKeepingVisualOperatorAction: Boolean by userDataOr { false }
|
|
||||||
internal var Editor.vimChangeActionSwitchMode: Mode? by userData()
|
internal var Editor.vimChangeActionSwitchMode: Mode? by userData()
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@@ -23,7 +23,7 @@ internal class VimStandalonePluginUpdateChecker : StandalonePluginUpdateChecker(
|
|||||||
VimIcons.IDEAVIM,
|
VimIcons.IDEAVIM,
|
||||||
) {
|
) {
|
||||||
|
|
||||||
override fun skipUpdateCheck(): Boolean = !VimPlugin.isEnabled() || "dev" in VimPlugin.getVersion()
|
override fun skipUpdateCheck(): Boolean = VimPlugin.isNotEnabled() || "dev" in VimPlugin.getVersion()
|
||||||
|
|
||||||
companion object {
|
companion object {
|
||||||
private const val PROPERTY_NAME = "ideavim.statistics.timestamp"
|
private const val PROPERTY_NAME = "ideavim.statistics.timestamp"
|
||||||
|
@@ -0,0 +1,25 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2003-2024 The IdeaVim authors
|
||||||
|
*
|
||||||
|
* Use of this source code is governed by an MIT-style
|
||||||
|
* license that can be found in the LICENSE.txt file or at
|
||||||
|
* https://opensource.org/licenses/MIT.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package com.maddyhome.idea.vim.ide
|
||||||
|
|
||||||
|
import com.intellij.openapi.extensions.ExtensionPointName
|
||||||
|
|
||||||
|
internal val clionEP = ExtensionPointName.create<ClionNovaProvider>("IdeaVIM.clionNovaProvider")
|
||||||
|
|
||||||
|
internal interface ClionNovaProvider {
|
||||||
|
fun isClionNova(): Boolean
|
||||||
|
}
|
||||||
|
|
||||||
|
internal class ClionNovaProviderImpl : ClionNovaProvider {
|
||||||
|
override fun isClionNova(): Boolean = true
|
||||||
|
}
|
||||||
|
|
||||||
|
internal fun isClionNova(): Boolean {
|
||||||
|
return clionEP.extensions.any { it.isClionNova() }
|
||||||
|
}
|
@@ -40,7 +40,7 @@ internal object AppCodeTemplates {
|
|||||||
private var editor: Editor? = null
|
private var editor: Editor? = null
|
||||||
|
|
||||||
override fun beforeActionPerformed(action: AnAction, event: AnActionEvent) {
|
override fun beforeActionPerformed(action: AnAction, event: AnActionEvent) {
|
||||||
if (!VimPlugin.isEnabled()) return
|
if (VimPlugin.isNotEnabled()) return
|
||||||
|
|
||||||
val hostEditor = event.dataContext.getData(CommonDataKeys.HOST_EDITOR)
|
val hostEditor = event.dataContext.getData(CommonDataKeys.HOST_EDITOR)
|
||||||
if (hostEditor != null) {
|
if (hostEditor != null) {
|
||||||
@@ -49,7 +49,7 @@ internal object AppCodeTemplates {
|
|||||||
}
|
}
|
||||||
|
|
||||||
override fun afterActionPerformed(action: AnAction, event: AnActionEvent, result: AnActionResult) {
|
override fun afterActionPerformed(action: AnAction, event: AnActionEvent, result: AnActionResult) {
|
||||||
if (!VimPlugin.isEnabled()) return
|
if (VimPlugin.isNotEnabled()) return
|
||||||
|
|
||||||
if (ActionManager.getInstance().getId(action) == IdeActions.ACTION_CHOOSE_LOOKUP_ITEM) {
|
if (ActionManager.getInstance().getId(action) == IdeActions.ACTION_CHOOSE_LOOKUP_ITEM) {
|
||||||
val myEditor = editor
|
val myEditor = editor
|
||||||
|
@@ -59,7 +59,7 @@ internal object IdeaSpecifics {
|
|||||||
private var completionPrevDocumentLength: Int? = null
|
private var completionPrevDocumentLength: Int? = null
|
||||||
private var completionPrevDocumentOffset: Int? = null
|
private var completionPrevDocumentOffset: Int? = null
|
||||||
override fun beforeActionPerformed(action: AnAction, event: AnActionEvent) {
|
override fun beforeActionPerformed(action: AnAction, event: AnActionEvent) {
|
||||||
if (!VimPlugin.isEnabled()) return
|
if (VimPlugin.isNotEnabled()) return
|
||||||
|
|
||||||
val hostEditor = event.dataContext.getData(CommonDataKeys.HOST_EDITOR)
|
val hostEditor = event.dataContext.getData(CommonDataKeys.HOST_EDITOR)
|
||||||
if (hostEditor != null) {
|
if (hostEditor != null) {
|
||||||
@@ -92,7 +92,7 @@ internal object IdeaSpecifics {
|
|||||||
}
|
}
|
||||||
|
|
||||||
override fun afterActionPerformed(action: AnAction, event: AnActionEvent, result: AnActionResult) {
|
override fun afterActionPerformed(action: AnAction, event: AnActionEvent, result: AnActionResult) {
|
||||||
if (!VimPlugin.isEnabled()) return
|
if (VimPlugin.isNotEnabled()) return
|
||||||
|
|
||||||
val editor = editor
|
val editor = editor
|
||||||
if (editor != null && action is ChooseItemAction && editor.vimStateMachine?.isRecording == true) {
|
if (editor != null && action is ChooseItemAction && editor.vimStateMachine?.isRecording == true) {
|
||||||
@@ -138,7 +138,7 @@ internal object IdeaSpecifics {
|
|||||||
//region Enter insert mode for surround templates without selection
|
//region Enter insert mode for surround templates without selection
|
||||||
class VimTemplateManagerListener : TemplateManagerListener {
|
class VimTemplateManagerListener : TemplateManagerListener {
|
||||||
override fun templateStarted(state: TemplateState) {
|
override fun templateStarted(state: TemplateState) {
|
||||||
if (!VimPlugin.isEnabled()) return
|
if (VimPlugin.isNotEnabled()) return
|
||||||
val editor = state.editor ?: return
|
val editor = state.editor ?: return
|
||||||
|
|
||||||
state.addTemplateStateListener(object : TemplateEditingAdapter() {
|
state.addTemplateStateListener(object : TemplateEditingAdapter() {
|
||||||
@@ -176,7 +176,7 @@ internal object IdeaSpecifics {
|
|||||||
//region Register shortcuts for lookup and perform partial reset
|
//region Register shortcuts for lookup and perform partial reset
|
||||||
class LookupTopicListener : LookupManagerListener {
|
class LookupTopicListener : LookupManagerListener {
|
||||||
override fun activeLookupChanged(oldLookup: Lookup?, newLookup: Lookup?) {
|
override fun activeLookupChanged(oldLookup: Lookup?, newLookup: Lookup?) {
|
||||||
if (!VimPlugin.isEnabled()) return
|
if (VimPlugin.isNotEnabled()) return
|
||||||
|
|
||||||
// Lookup opened
|
// Lookup opened
|
||||||
if (oldLookup == null && newLookup is LookupImpl) {
|
if (oldLookup == null && newLookup is LookupImpl) {
|
||||||
@@ -199,7 +199,7 @@ internal object IdeaSpecifics {
|
|||||||
//region Hide Vim search highlights when showing IntelliJ search results
|
//region Hide Vim search highlights when showing IntelliJ search results
|
||||||
class VimFindModelListener : FindModelListener {
|
class VimFindModelListener : FindModelListener {
|
||||||
override fun findNextModelChanged() {
|
override fun findNextModelChanged() {
|
||||||
if (!VimPlugin.isEnabled()) return
|
if (VimPlugin.isNotEnabled()) return
|
||||||
VimPlugin.getSearch().clearSearchHighlight()
|
VimPlugin.getSearch().clearSearchHighlight()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -27,7 +27,7 @@ internal class RiderActionListener : AnActionListener {
|
|||||||
|
|
||||||
private var editor: Editor? = null
|
private var editor: Editor? = null
|
||||||
override fun beforeActionPerformed(action: AnAction, event: AnActionEvent) {
|
override fun beforeActionPerformed(action: AnAction, event: AnActionEvent) {
|
||||||
if (!VimPlugin.isEnabled()) return
|
if (VimPlugin.isNotEnabled()) return
|
||||||
|
|
||||||
val hostEditor = event.dataContext.getData(CommonDataKeys.HOST_EDITOR)
|
val hostEditor = event.dataContext.getData(CommonDataKeys.HOST_EDITOR)
|
||||||
if (hostEditor != null) {
|
if (hostEditor != null) {
|
||||||
@@ -36,7 +36,7 @@ internal class RiderActionListener : AnActionListener {
|
|||||||
}
|
}
|
||||||
|
|
||||||
override fun afterActionPerformed(action: AnAction, event: AnActionEvent, result: AnActionResult) {
|
override fun afterActionPerformed(action: AnAction, event: AnActionEvent, result: AnActionResult) {
|
||||||
if (!VimPlugin.isEnabled()) return
|
if (VimPlugin.isNotEnabled()) return
|
||||||
|
|
||||||
//region Extend Selection for Rider
|
//region Extend Selection for Rider
|
||||||
when (ActionManager.getInstance().getId(action)) {
|
when (ActionManager.getInstance().getId(action)) {
|
||||||
|
@@ -9,6 +9,7 @@
|
|||||||
package com.maddyhome.idea.vim.listener
|
package com.maddyhome.idea.vim.listener
|
||||||
|
|
||||||
import com.intellij.ide.ui.UISettings
|
import com.intellij.ide.ui.UISettings
|
||||||
|
import com.intellij.openapi.Disposable
|
||||||
import com.intellij.openapi.application.ApplicationManager
|
import com.intellij.openapi.application.ApplicationManager
|
||||||
import com.intellij.openapi.diagnostic.Logger
|
import com.intellij.openapi.diagnostic.Logger
|
||||||
import com.intellij.openapi.diagnostic.trace
|
import com.intellij.openapi.diagnostic.trace
|
||||||
@@ -28,8 +29,9 @@ import com.intellij.openapi.editor.event.EditorMouseMotionListener
|
|||||||
import com.intellij.openapi.editor.event.SelectionEvent
|
import com.intellij.openapi.editor.event.SelectionEvent
|
||||||
import com.intellij.openapi.editor.event.SelectionListener
|
import com.intellij.openapi.editor.event.SelectionListener
|
||||||
import com.intellij.openapi.editor.ex.DocumentEx
|
import com.intellij.openapi.editor.ex.DocumentEx
|
||||||
|
import com.intellij.openapi.editor.ex.EditorEventMulticasterEx
|
||||||
|
import com.intellij.openapi.editor.ex.FocusChangeListener
|
||||||
import com.intellij.openapi.editor.impl.EditorComponentImpl
|
import com.intellij.openapi.editor.impl.EditorComponentImpl
|
||||||
import com.intellij.openapi.editor.impl.EditorImpl
|
|
||||||
import com.intellij.openapi.fileEditor.FileEditorManager
|
import com.intellij.openapi.fileEditor.FileEditorManager
|
||||||
import com.intellij.openapi.fileEditor.FileEditorManagerEvent
|
import com.intellij.openapi.fileEditor.FileEditorManagerEvent
|
||||||
import com.intellij.openapi.fileEditor.FileEditorManagerListener
|
import com.intellij.openapi.fileEditor.FileEditorManagerListener
|
||||||
@@ -40,14 +42,11 @@ import com.intellij.openapi.fileEditor.ex.FileEditorWithProvider
|
|||||||
import com.intellij.openapi.fileEditor.impl.EditorComposite
|
import com.intellij.openapi.fileEditor.impl.EditorComposite
|
||||||
import com.intellij.openapi.fileEditor.impl.EditorWindow
|
import com.intellij.openapi.fileEditor.impl.EditorWindow
|
||||||
import com.intellij.openapi.project.ProjectManager
|
import com.intellij.openapi.project.ProjectManager
|
||||||
import com.intellij.openapi.rd.createLifetime
|
|
||||||
import com.intellij.openapi.rd.createNestedDisposable
|
|
||||||
import com.intellij.openapi.util.Disposer
|
import com.intellij.openapi.util.Disposer
|
||||||
import com.intellij.openapi.util.Key
|
import com.intellij.openapi.util.Key
|
||||||
import com.intellij.openapi.util.removeUserData
|
import com.intellij.openapi.util.removeUserData
|
||||||
import com.intellij.openapi.vfs.VirtualFile
|
import com.intellij.openapi.vfs.VirtualFile
|
||||||
import com.intellij.util.ExceptionUtil
|
import com.intellij.util.ExceptionUtil
|
||||||
import com.jetbrains.rd.util.lifetime.Lifetime
|
|
||||||
import com.maddyhome.idea.vim.EventFacade
|
import com.maddyhome.idea.vim.EventFacade
|
||||||
import com.maddyhome.idea.vim.KeyHandler
|
import com.maddyhome.idea.vim.KeyHandler
|
||||||
import com.maddyhome.idea.vim.VimKeyListener
|
import com.maddyhome.idea.vim.VimKeyListener
|
||||||
@@ -56,12 +55,14 @@ import com.maddyhome.idea.vim.VimTypedActionHandler
|
|||||||
import com.maddyhome.idea.vim.api.LocalOptionInitialisationScenario
|
import com.maddyhome.idea.vim.api.LocalOptionInitialisationScenario
|
||||||
import com.maddyhome.idea.vim.api.Options
|
import com.maddyhome.idea.vim.api.Options
|
||||||
import com.maddyhome.idea.vim.api.VimEditor
|
import com.maddyhome.idea.vim.api.VimEditor
|
||||||
|
import com.maddyhome.idea.vim.api.coerceOffset
|
||||||
import com.maddyhome.idea.vim.api.getLineEndForOffset
|
import com.maddyhome.idea.vim.api.getLineEndForOffset
|
||||||
import com.maddyhome.idea.vim.api.getLineStartForOffset
|
import com.maddyhome.idea.vim.api.getLineStartForOffset
|
||||||
import com.maddyhome.idea.vim.api.injector
|
import com.maddyhome.idea.vim.api.injector
|
||||||
import com.maddyhome.idea.vim.ex.ExOutputModel
|
import com.maddyhome.idea.vim.ex.ExOutputModel
|
||||||
import com.maddyhome.idea.vim.group.EditorGroup
|
import com.maddyhome.idea.vim.group.EditorGroup
|
||||||
import com.maddyhome.idea.vim.group.FileGroup
|
import com.maddyhome.idea.vim.group.FileGroup
|
||||||
|
import com.maddyhome.idea.vim.group.IjOptions
|
||||||
import com.maddyhome.idea.vim.group.MotionGroup
|
import com.maddyhome.idea.vim.group.MotionGroup
|
||||||
import com.maddyhome.idea.vim.group.OptionGroup
|
import com.maddyhome.idea.vim.group.OptionGroup
|
||||||
import com.maddyhome.idea.vim.group.ScrollGroup
|
import com.maddyhome.idea.vim.group.ScrollGroup
|
||||||
@@ -70,6 +71,8 @@ import com.maddyhome.idea.vim.group.visual.IdeaSelectionControl
|
|||||||
import com.maddyhome.idea.vim.group.visual.VimVisualTimer
|
import com.maddyhome.idea.vim.group.visual.VimVisualTimer
|
||||||
import com.maddyhome.idea.vim.group.visual.moveCaretOneCharLeftFromSelectionEnd
|
import com.maddyhome.idea.vim.group.visual.moveCaretOneCharLeftFromSelectionEnd
|
||||||
import com.maddyhome.idea.vim.group.visual.vimSetSystemSelectionSilently
|
import com.maddyhome.idea.vim.group.visual.vimSetSystemSelectionSilently
|
||||||
|
import com.maddyhome.idea.vim.handler.correctorRequester
|
||||||
|
import com.maddyhome.idea.vim.handler.keyCheckRequests
|
||||||
import com.maddyhome.idea.vim.helper.GuicursorChangeListener
|
import com.maddyhome.idea.vim.helper.GuicursorChangeListener
|
||||||
import com.maddyhome.idea.vim.helper.StrictMode
|
import com.maddyhome.idea.vim.helper.StrictMode
|
||||||
import com.maddyhome.idea.vim.helper.VimStandalonePluginUpdateChecker
|
import com.maddyhome.idea.vim.helper.VimStandalonePluginUpdateChecker
|
||||||
@@ -94,6 +97,9 @@ import com.maddyhome.idea.vim.state.mode.mode
|
|||||||
import com.maddyhome.idea.vim.state.mode.selectionType
|
import com.maddyhome.idea.vim.state.mode.selectionType
|
||||||
import com.maddyhome.idea.vim.ui.ShowCmdOptionChangeListener
|
import com.maddyhome.idea.vim.ui.ShowCmdOptionChangeListener
|
||||||
import com.maddyhome.idea.vim.ui.ex.ExEntryPanel
|
import com.maddyhome.idea.vim.ui.ex.ExEntryPanel
|
||||||
|
import com.maddyhome.idea.vim.ui.widgets.macro.macroWidgetOptionListener
|
||||||
|
import com.maddyhome.idea.vim.ui.widgets.mode.modeWidgetOptionListener
|
||||||
|
import com.maddyhome.idea.vim.vimDisposable
|
||||||
import java.awt.event.MouseAdapter
|
import java.awt.event.MouseAdapter
|
||||||
import java.awt.event.MouseEvent
|
import java.awt.event.MouseEvent
|
||||||
import javax.swing.SwingUtilities
|
import javax.swing.SwingUtilities
|
||||||
@@ -128,11 +134,14 @@ internal object VimListenerManager {
|
|||||||
fun turnOn() {
|
fun turnOn() {
|
||||||
GlobalListeners.enable()
|
GlobalListeners.enable()
|
||||||
EditorListeners.addAll()
|
EditorListeners.addAll()
|
||||||
|
check(correctorRequester.tryEmit(Unit))
|
||||||
|
check(keyCheckRequests.tryEmit(Unit))
|
||||||
}
|
}
|
||||||
|
|
||||||
fun turnOff() {
|
fun turnOff() {
|
||||||
GlobalListeners.disable()
|
GlobalListeners.disable()
|
||||||
EditorListeners.removeAll()
|
EditorListeners.removeAll()
|
||||||
|
check(correctorRequester.tryEmit(Unit))
|
||||||
}
|
}
|
||||||
|
|
||||||
object GlobalListeners {
|
object GlobalListeners {
|
||||||
@@ -150,6 +159,13 @@ internal object VimListenerManager {
|
|||||||
optionGroup.addEffectiveOptionValueChangeListener(Options.relativenumber, EditorGroup.NumberChangeListener.INSTANCE)
|
optionGroup.addEffectiveOptionValueChangeListener(Options.relativenumber, EditorGroup.NumberChangeListener.INSTANCE)
|
||||||
optionGroup.addEffectiveOptionValueChangeListener(Options.scrolloff, ScrollGroup.ScrollOptionsChangeListener)
|
optionGroup.addEffectiveOptionValueChangeListener(Options.scrolloff, ScrollGroup.ScrollOptionsChangeListener)
|
||||||
optionGroup.addGlobalOptionChangeListener(Options.showcmd, ShowCmdOptionChangeListener)
|
optionGroup.addGlobalOptionChangeListener(Options.showcmd, ShowCmdOptionChangeListener)
|
||||||
|
|
||||||
|
// This code is executed after ideavimrc execution, so we trigger onGlobalOptionChanged just in case
|
||||||
|
optionGroup.addGlobalOptionChangeListener(IjOptions.showmodewidget, modeWidgetOptionListener)
|
||||||
|
optionGroup.addGlobalOptionChangeListener(IjOptions.showmodewidget, macroWidgetOptionListener)
|
||||||
|
modeWidgetOptionListener.onGlobalOptionChanged()
|
||||||
|
macroWidgetOptionListener.onGlobalOptionChanged()
|
||||||
|
|
||||||
optionGroup.addEffectiveOptionValueChangeListener(Options.guicursor, GuicursorChangeListener)
|
optionGroup.addEffectiveOptionValueChangeListener(Options.guicursor, GuicursorChangeListener)
|
||||||
|
|
||||||
EventFacade.getInstance().addEditorFactoryListener(VimEditorFactoryListener, VimPlugin.getInstance().onOffDisposable)
|
EventFacade.getInstance().addEditorFactoryListener(VimEditorFactoryListener, VimPlugin.getInstance().onOffDisposable)
|
||||||
@@ -157,6 +173,8 @@ internal object VimListenerManager {
|
|||||||
busConnection.subscribe(FileOpenedSyncListener.TOPIC, VimEditorFactoryListener)
|
busConnection.subscribe(FileOpenedSyncListener.TOPIC, VimEditorFactoryListener)
|
||||||
|
|
||||||
EditorFactory.getInstance().eventMulticaster.addCaretListener(VimCaretListener, VimPlugin.getInstance().onOffDisposable)
|
EditorFactory.getInstance().eventMulticaster.addCaretListener(VimCaretListener, VimPlugin.getInstance().onOffDisposable)
|
||||||
|
val eventMulticaster = EditorFactory.getInstance().eventMulticaster as? EditorEventMulticasterEx
|
||||||
|
eventMulticaster?.addFocusChangeListener(VimFocusListener, VimPlugin.getInstance().onOffDisposable)
|
||||||
}
|
}
|
||||||
|
|
||||||
fun disable() {
|
fun disable() {
|
||||||
@@ -167,6 +185,8 @@ internal object VimListenerManager {
|
|||||||
optionGroup.removeEffectiveOptionValueChangeListener(Options.relativenumber, EditorGroup.NumberChangeListener.INSTANCE)
|
optionGroup.removeEffectiveOptionValueChangeListener(Options.relativenumber, EditorGroup.NumberChangeListener.INSTANCE)
|
||||||
optionGroup.removeEffectiveOptionValueChangeListener(Options.scrolloff, ScrollGroup.ScrollOptionsChangeListener)
|
optionGroup.removeEffectiveOptionValueChangeListener(Options.scrolloff, ScrollGroup.ScrollOptionsChangeListener)
|
||||||
optionGroup.removeGlobalOptionChangeListener(Options.showcmd, ShowCmdOptionChangeListener)
|
optionGroup.removeGlobalOptionChangeListener(Options.showcmd, ShowCmdOptionChangeListener)
|
||||||
|
optionGroup.removeGlobalOptionChangeListener(IjOptions.showmodewidget, modeWidgetOptionListener)
|
||||||
|
optionGroup.removeGlobalOptionChangeListener(IjOptions.showmodewidget, macroWidgetOptionListener)
|
||||||
optionGroup.removeEffectiveOptionValueChangeListener(Options.guicursor, GuicursorChangeListener)
|
optionGroup.removeEffectiveOptionValueChangeListener(Options.guicursor, GuicursorChangeListener)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -209,49 +229,67 @@ internal object VimListenerManager {
|
|||||||
}
|
}
|
||||||
|
|
||||||
fun add(editor: Editor, openingEditor: VimEditor, scenario: LocalOptionInitialisationScenario) {
|
fun add(editor: Editor, openingEditor: VimEditor, scenario: LocalOptionInitialisationScenario) {
|
||||||
val pluginLifetime = VimPlugin.getInstance().createLifetime()
|
// As I understand, there is no need to pass a disposable that also disposes on editor close
|
||||||
val editorLifetime = (editor as EditorImpl).disposable.createLifetime()
|
// because all editor resources will be garbage collected anyway on editor close
|
||||||
val disposable =
|
val disposable = editor.project?.vimDisposable ?: return
|
||||||
Lifetime.intersect(pluginLifetime, editorLifetime).createNestedDisposable("MyLifetimedDisposable")
|
|
||||||
|
val listenersDisposable = Disposer.newDisposable(disposable)
|
||||||
|
editor.putUserData(editorListenersDisposable, listenersDisposable)
|
||||||
|
|
||||||
|
Disposer.register(listenersDisposable) {
|
||||||
|
if (VimListenerTestObject.enabled) {
|
||||||
|
VimListenerTestObject.disposedCounter += 1
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
editor.contentComponent.addKeyListener(VimKeyListener)
|
editor.contentComponent.addKeyListener(VimKeyListener)
|
||||||
Disposer.register(disposable) { editor.contentComponent.removeKeyListener(VimKeyListener) }
|
Disposer.register(listenersDisposable) { editor.contentComponent.removeKeyListener(VimKeyListener) }
|
||||||
|
|
||||||
// Initialise the local options. We MUST do this before anything has the chance to query options
|
// Initialise the local options. We MUST do this before anything has the chance to query options
|
||||||
VimPlugin.getOptionGroup().initialiseLocalOptions(editor.vim, openingEditor, scenario)
|
val vimEditor = editor.vim
|
||||||
|
VimPlugin.getOptionGroup().initialiseLocalOptions(vimEditor, openingEditor, scenario)
|
||||||
|
|
||||||
val eventFacade = EventFacade.getInstance()
|
val eventFacade = EventFacade.getInstance()
|
||||||
eventFacade.addEditorMouseListener(editor, EditorMouseHandler, disposable)
|
eventFacade.addEditorMouseListener(editor, EditorMouseHandler, listenersDisposable)
|
||||||
eventFacade.addEditorMouseMotionListener(editor, EditorMouseHandler, disposable)
|
eventFacade.addEditorMouseMotionListener(editor, EditorMouseHandler, listenersDisposable)
|
||||||
eventFacade.addEditorSelectionListener(editor, EditorSelectionHandler, disposable)
|
eventFacade.addEditorSelectionListener(editor, EditorSelectionHandler, listenersDisposable)
|
||||||
eventFacade.addComponentMouseListener(editor.contentComponent, ComponentMouseListener, disposable)
|
eventFacade.addComponentMouseListener(editor.contentComponent, ComponentMouseListener, listenersDisposable)
|
||||||
eventFacade.addCaretListener(editor, EditorCaretHandler, disposable)
|
eventFacade.addCaretListener(editor, EditorCaretHandler, listenersDisposable)
|
||||||
|
|
||||||
VimPlugin.getEditor().editorCreated(editor)
|
VimPlugin.getEditor().editorCreated(editor)
|
||||||
|
|
||||||
VimPlugin.getChange().editorCreated(editor, disposable)
|
VimPlugin.getChange().editorCreated(editor, listenersDisposable)
|
||||||
|
|
||||||
Disposer.register(disposable) {
|
injector.listenersNotifier.notifyEditorCreated(vimEditor)
|
||||||
|
|
||||||
|
Disposer.register(listenersDisposable) {
|
||||||
VimPlugin.getEditorIfCreated()?.editorDeinit(editor, true)
|
VimPlugin.getEditorIfCreated()?.editorDeinit(editor, true)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fun remove(editor: Editor, isReleased: Boolean) {
|
fun remove(editor: Editor, isReleased: Boolean) {
|
||||||
editor.contentComponent.removeKeyListener(VimKeyListener)
|
val editorDisposable = editor.getUserData(editorListenersDisposable)
|
||||||
val eventFacade = EventFacade.getInstance()
|
if (editorDisposable != null) {
|
||||||
eventFacade.removeEditorMouseListener(editor, EditorMouseHandler)
|
Disposer.dispose(editorDisposable)
|
||||||
eventFacade.removeEditorMouseMotionListener(editor, EditorMouseHandler)
|
}
|
||||||
eventFacade.removeEditorSelectionListener(editor, EditorSelectionHandler)
|
else StrictMode.fail("Editor doesn't have disposable attached. $editor")
|
||||||
eventFacade.removeComponentMouseListener(editor.contentComponent, ComponentMouseListener)
|
|
||||||
eventFacade.removeCaretListener(editor, EditorCaretHandler)
|
|
||||||
|
|
||||||
VimPlugin.getEditorIfCreated()?.editorDeinit(editor, isReleased)
|
VimPlugin.getEditorIfCreated()?.editorDeinit(editor, isReleased)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
VimPlugin.getChange().editorReleased(editor)
|
private object VimFocusListener : FocusChangeListener {
|
||||||
|
override fun focusGained(editor: Editor) {
|
||||||
|
injector.listenersNotifier.notifyEditorFocusGained(editor.vim)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
override fun focusLost(editor: Editor) {
|
||||||
|
injector.listenersNotifier.notifyEditorFocusLost(editor.vim)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
val editorListenersDisposable = Key.create<Disposable>("IdeaVim listeners disposable")
|
||||||
|
|
||||||
object VimCaretListener : CaretListener {
|
object VimCaretListener : CaretListener {
|
||||||
override fun caretAdded(event: CaretEvent) {
|
override fun caretAdded(event: CaretEvent) {
|
||||||
if (vimDisabled(event.editor)) return
|
if (vimDisabled(event.editor)) return
|
||||||
@@ -266,10 +304,10 @@ internal object VimListenerManager {
|
|||||||
|
|
||||||
class VimFileEditorManagerListener : FileEditorManagerListener {
|
class VimFileEditorManagerListener : FileEditorManagerListener {
|
||||||
override fun selectionChanged(event: FileEditorManagerEvent) {
|
override fun selectionChanged(event: FileEditorManagerEvent) {
|
||||||
if (!VimPlugin.isEnabled()) return
|
if (VimPlugin.isNotEnabled()) return
|
||||||
MotionGroup.fileEditorManagerSelectionChangedCallback(event)
|
MotionGroup.fileEditorManagerSelectionChangedCallback(event)
|
||||||
FileGroup.fileEditorManagerSelectionChangedCallback(event)
|
FileGroup.fileEditorManagerSelectionChangedCallback(event)
|
||||||
SearchGroup.fileEditorManagerSelectionChangedCallback(event)
|
VimPlugin.getSearch().fileEditorManagerSelectionChangedCallback(event)
|
||||||
OptionGroup.fileEditorManagerSelectionChangedCallback(event)
|
OptionGroup.fileEditorManagerSelectionChangedCallback(event)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -336,13 +374,15 @@ internal object VimListenerManager {
|
|||||||
}
|
}
|
||||||
|
|
||||||
override fun editorReleased(event: EditorFactoryEvent) {
|
override fun editorReleased(event: EditorFactoryEvent) {
|
||||||
injector.markService.editorReleased(event.editor.vim)
|
val vimEditor = event.editor.vim
|
||||||
|
injector.listenersNotifier.notifyEditorReleased(vimEditor)
|
||||||
|
injector.markService.editorReleased(vimEditor)
|
||||||
}
|
}
|
||||||
|
|
||||||
override fun fileOpenedSync(
|
override fun fileOpenedSync(
|
||||||
source: FileEditorManager,
|
source: FileEditorManager,
|
||||||
file: VirtualFile,
|
file: VirtualFile,
|
||||||
editorsWithProviders: List<FileEditorWithProvider>
|
editorsWithProviders: List<FileEditorWithProvider>,
|
||||||
) {
|
) {
|
||||||
// This callback is called once all editors are created for a file being opened. The EditorComposite has been
|
// This callback is called once all editors are created for a file being opened. The EditorComposite has been
|
||||||
// created (and the list of editors and providers is passed here) and added to an EditorWindow tab, inside a
|
// created (and the list of editors and providers is passed here) and added to an EditorWindow tab, inside a
|
||||||
@@ -400,6 +440,7 @@ internal object VimListenerManager {
|
|||||||
*/
|
*/
|
||||||
override fun selectionChanged(selectionEvent: SelectionEvent) {
|
override fun selectionChanged(selectionEvent: SelectionEvent) {
|
||||||
if (selectionEvent.editor.isIdeaVimDisabledHere) return
|
if (selectionEvent.editor.isIdeaVimDisabledHere) return
|
||||||
|
VimVisualTimer.drop()
|
||||||
val editor = selectionEvent.editor
|
val editor = selectionEvent.editor
|
||||||
val document = editor.document
|
val document = editor.document
|
||||||
val ijVimEditor = IjVimEditor(editor)
|
val ijVimEditor = IjVimEditor(editor)
|
||||||
@@ -422,11 +463,17 @@ internal object VimListenerManager {
|
|||||||
if (lineEnd == endOffset - 1) {
|
if (lineEnd == endOffset - 1) {
|
||||||
// When starting on an empty line and dragging vertically upwards onto
|
// When starting on an empty line and dragging vertically upwards onto
|
||||||
// another line, the selection should include the entirety of the empty line
|
// another line, the selection should include the entirety of the empty line
|
||||||
caret.setSelection(endOffset + 1, startOffset)
|
caret.setSelection(
|
||||||
|
ijVimEditor.coerceOffset(endOffset + 1).point,
|
||||||
|
ijVimEditor.coerceOffset(startOffset).point,
|
||||||
|
)
|
||||||
} else if (lineEnd == startOffset + 1 && startOffset == endOffset) {
|
} else if (lineEnd == startOffset + 1 && startOffset == endOffset) {
|
||||||
// When dragging left from EOL on a non-empty line, the selection
|
// When dragging left from EOL on a non-empty line, the selection
|
||||||
// should include the last character on the line
|
// should include the last character on the line
|
||||||
caret.setSelection(lineEnd, lineEnd - 1)
|
caret.setSelection(
|
||||||
|
ijVimEditor.coerceOffset(lineEnd).point,
|
||||||
|
ijVimEditor.coerceOffset(lineEnd - 1).point,
|
||||||
|
)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
//endregion
|
//endregion
|
||||||
@@ -693,6 +740,11 @@ internal object VimListenerManager {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
internal object VimListenerTestObject {
|
||||||
|
var enabled: Boolean = false
|
||||||
|
var disposedCounter = 0
|
||||||
|
}
|
||||||
|
|
||||||
private object MouseEventsDataHolder {
|
private object MouseEventsDataHolder {
|
||||||
const val skipEvents = 3
|
const val skipEvents = 3
|
||||||
var skipNDragEvents = skipEvents
|
var skipNDragEvents = skipEvents
|
||||||
|
@@ -18,6 +18,7 @@ import com.intellij.openapi.editor.CaretStateTransferableData
|
|||||||
import com.intellij.openapi.editor.RawText
|
import com.intellij.openapi.editor.RawText
|
||||||
import com.intellij.openapi.editor.richcopy.view.HtmlTransferableData
|
import com.intellij.openapi.editor.richcopy.view.HtmlTransferableData
|
||||||
import com.intellij.openapi.editor.richcopy.view.RtfTransferableData
|
import com.intellij.openapi.editor.richcopy.view.RtfTransferableData
|
||||||
|
import com.intellij.openapi.project.DumbService
|
||||||
import com.intellij.openapi.project.IndexNotReadyException
|
import com.intellij.openapi.project.IndexNotReadyException
|
||||||
import com.intellij.psi.PsiDocumentManager
|
import com.intellij.psi.PsiDocumentManager
|
||||||
import com.intellij.util.ui.EmptyClipboardOwner
|
import com.intellij.util.ui.EmptyClipboardOwner
|
||||||
@@ -100,8 +101,7 @@ internal class IjClipboardManager : VimClipboardManager {
|
|||||||
|
|
||||||
// This thing enables alternative context resolve for dumb mode.
|
// This thing enables alternative context resolve for dumb mode.
|
||||||
// Please read docs for com.intellij.openapi.project.DumbService.isAlternativeResolveEnabled
|
// Please read docs for com.intellij.openapi.project.DumbService.isAlternativeResolveEnabled
|
||||||
// [VERSION UPDATE] 2023.2+ Enable alternative context back
|
DumbService.getInstance(project).withAlternativeResolveEnabled {
|
||||||
// DumbService.getInstance(project).withAlternativeResolveEnabled {
|
|
||||||
for (processor in CopyPastePostProcessor.EP_NAME.extensionList) {
|
for (processor in CopyPastePostProcessor.EP_NAME.extensionList) {
|
||||||
try {
|
try {
|
||||||
logger.debug { "Copy paste processor: ${processor.javaClass.name}" }
|
logger.debug { "Copy paste processor: ${processor.javaClass.name}" }
|
||||||
@@ -116,7 +116,7 @@ internal class IjClipboardManager : VimClipboardManager {
|
|||||||
} catch (ignore: IndexNotReadyException) {
|
} catch (ignore: IndexNotReadyException) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
// }
|
}
|
||||||
transferableData.add(CaretStateTransferableData(intArrayOf(0), intArrayOf(text.length)))
|
transferableData.add(CaretStateTransferableData(intArrayOf(0), intArrayOf(text.length)))
|
||||||
|
|
||||||
// These data provided by {@link com.intellij.openapi.editor.richcopy.TextWithMarkupProcessor} doesn't work with
|
// These data provided by {@link com.intellij.openapi.editor.richcopy.TextWithMarkupProcessor} doesn't work with
|
||||||
|
@@ -1,23 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright 2003-2023 The IdeaVim authors
|
|
||||||
*
|
|
||||||
* Use of this source code is governed by an MIT-style
|
|
||||||
* license that can be found in the LICENSE.txt file or at
|
|
||||||
* https://opensource.org/licenses/MIT.
|
|
||||||
*/
|
|
||||||
|
|
||||||
package com.maddyhome.idea.vim.newapi
|
|
||||||
|
|
||||||
import com.maddyhome.idea.vim.api.VimActionsInitiator
|
|
||||||
import com.maddyhome.idea.vim.handler.ActionBeanClass
|
|
||||||
import com.maddyhome.idea.vim.handler.EditorActionHandlerBase
|
|
||||||
import org.jetbrains.annotations.ApiStatus
|
|
||||||
|
|
||||||
@Deprecated(message = "Please use CommandOrMotion annotation")
|
|
||||||
@ApiStatus.ScheduledForRemoval(inVersion = "2.9.0")
|
|
||||||
internal class IjVimActionsInitiator(val bean: ActionBeanClass) : VimActionsInitiator {
|
|
||||||
override fun getInstance(): EditorActionHandlerBase = bean.instance
|
|
||||||
}
|
|
||||||
|
|
||||||
internal val VimActionsInitiator.ij: ActionBeanClass
|
|
||||||
get() = (this as IjVimActionsInitiator).bean
|
|
@@ -54,7 +54,6 @@ import com.maddyhome.idea.vim.helper.isTemplateActive
|
|||||||
import com.maddyhome.idea.vim.helper.updateCaretsVisualAttributes
|
import com.maddyhome.idea.vim.helper.updateCaretsVisualAttributes
|
||||||
import com.maddyhome.idea.vim.helper.updateCaretsVisualPosition
|
import com.maddyhome.idea.vim.helper.updateCaretsVisualPosition
|
||||||
import com.maddyhome.idea.vim.helper.vimChangeActionSwitchMode
|
import com.maddyhome.idea.vim.helper.vimChangeActionSwitchMode
|
||||||
import com.maddyhome.idea.vim.helper.vimKeepingVisualOperatorAction
|
|
||||||
import com.maddyhome.idea.vim.helper.vimLastSelectionType
|
import com.maddyhome.idea.vim.helper.vimLastSelectionType
|
||||||
import com.maddyhome.idea.vim.state.mode.Mode
|
import com.maddyhome.idea.vim.state.mode.Mode
|
||||||
import com.maddyhome.idea.vim.state.mode.SelectionType
|
import com.maddyhome.idea.vim.state.mode.SelectionType
|
||||||
@@ -82,11 +81,6 @@ internal class IjVimEditor(editor: Editor) : MutableLinearEditor() {
|
|||||||
set(value) {
|
set(value) {
|
||||||
editor.vimChangeActionSwitchMode = value
|
editor.vimChangeActionSwitchMode = value
|
||||||
}
|
}
|
||||||
override var vimKeepingVisualOperatorAction: Boolean
|
|
||||||
get() = editor.vimKeepingVisualOperatorAction
|
|
||||||
set(value) {
|
|
||||||
editor.vimKeepingVisualOperatorAction = value
|
|
||||||
}
|
|
||||||
|
|
||||||
override fun fileSize(): Long = editor.fileSize.toLong()
|
override fun fileSize(): Long = editor.fileSize.toLong()
|
||||||
|
|
||||||
|
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user