mirror of
				https://github.com/chylex/IntelliJ-IdeaVim.git
				synced 2025-10-26 14:23:42 +01:00 
			
		
		
		
	Compare commits
	
		
			16 Commits
		
	
	
		
			a7def05aa8
			...
			customized
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 1afd0d3c37 | |||
| 547bd08c7e | |||
| 0c4154547f | |||
| 92d8be1ba4 | |||
| 080dbe9c0e | |||
| ff7e6bf053 | |||
| f19ce34311 | |||
| 8bcb0d116d | |||
| d9ae9fa40d | |||
| c19f88e5c0 | |||
| 03f4fb288d | |||
| ef27579277 | |||
| a8a822b58e | |||
| 9752bd5bbb | |||
| 240d9e0be4 | |||
| 328442544d | 
							
								
								
									
										1
									
								
								.github/workflows/checkNewPlugins.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								.github/workflows/checkNewPlugins.yml
									
									
									
									
										vendored
									
									
								
							| @@ -14,7 +14,6 @@ jobs: | |||||||
|   build: |   build: | ||||||
|  |  | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     if: github.repository == 'JetBrains/ideavim' |  | ||||||
|  |  | ||||||
|     steps: |     steps: | ||||||
|       - name: Fetch origin repo |       - name: Fetch origin repo | ||||||
|   | |||||||
							
								
								
									
										1
									
								
								.github/workflows/closeYoutrackOnCommit.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								.github/workflows/closeYoutrackOnCommit.yml
									
									
									
									
										vendored
									
									
								
							| @@ -12,7 +12,6 @@ jobs: | |||||||
|   build: |   build: | ||||||
|  |  | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     if: github.repository == 'JetBrains/ideavim' |  | ||||||
|  |  | ||||||
|     steps: |     steps: | ||||||
|       - uses: actions/checkout@v3 |       - uses: actions/checkout@v3 | ||||||
|   | |||||||
							
								
								
									
										1
									
								
								.github/workflows/integrationsTest.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								.github/workflows/integrationsTest.yml
									
									
									
									
										vendored
									
									
								
							| @@ -12,7 +12,6 @@ jobs: | |||||||
|   build: |   build: | ||||||
|  |  | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     if: github.repository == 'JetBrains/ideavim' |  | ||||||
|  |  | ||||||
|     steps: |     steps: | ||||||
|       - uses: actions/checkout@v2 |       - uses: actions/checkout@v2 | ||||||
|   | |||||||
							
								
								
									
										1
									
								
								.github/workflows/kover.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								.github/workflows/kover.yml
									
									
									
									
										vendored
									
									
								
							| @@ -12,7 +12,6 @@ jobs: | |||||||
|   build: |   build: | ||||||
|  |  | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     if: github.repository == 'JetBrains/ideavim' |  | ||||||
|  |  | ||||||
|     steps: |     steps: | ||||||
|       - uses: actions/checkout@v2 |       - uses: actions/checkout@v2 | ||||||
|   | |||||||
							
								
								
									
										2
									
								
								.github/workflows/mergeDependabotPR.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/mergeDependabotPR.yml
									
									
									
									
										vendored
									
									
								
							| @@ -8,7 +8,7 @@ permissions: | |||||||
| jobs: | jobs: | ||||||
|   dependabot: |   dependabot: | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     if: ${{ github.actor == 'dependabot[bot]' && github.repository == 'JetBrains/ideavim' }} |     if: ${{ github.actor == 'dependabot[bot]' }} | ||||||
|     steps: |     steps: | ||||||
|       - name: Auto-merge Dependabot PR |       - name: Auto-merge Dependabot PR | ||||||
|         run: gh pr merge --auto --rebase "$PR_URL" |         run: gh pr merge --auto --rebase "$PR_URL" | ||||||
|   | |||||||
							
								
								
									
										2
									
								
								.github/workflows/mergePr.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/mergePr.yml
									
									
									
									
										vendored
									
									
								
							| @@ -11,7 +11,7 @@ on: | |||||||
| jobs: | jobs: | ||||||
|   build: |   build: | ||||||
|  |  | ||||||
|     if: github.event.pull_request.merged == true && github.repository == 'JetBrains/ideavim' |     if: github.event.pull_request.merged == true | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|  |  | ||||||
|     steps: |     steps: | ||||||
|   | |||||||
							
								
								
									
										87
									
								
								.github/workflows/runUiOctopusTests.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										87
									
								
								.github/workflows/runUiOctopusTests.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,87 +0,0 @@ | |||||||
| 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 tests/ui-ij-tests/src/test/kotlin/ui/octopus.patch |  | ||||||
|       - name: Setup Java |  | ||||||
|         uses: actions/setup-java@v4 |  | ||||||
|         with: |  | ||||||
|           distribution: zulu |  | ||||||
|           java-version: 17 |  | ||||||
|       - 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 :tests:ui-ij-tests:testUi |  | ||||||
|       - name: Move video |  | ||||||
|         if: always() |  | ||||||
|         run: mv tests/ui-ij-tests/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 |  | ||||||
							
								
								
									
										55
									
								
								.github/workflows/runUiPyTests.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										55
									
								
								.github/workflows/runUiPyTests.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,55 +0,0 @@ | |||||||
| name: Run UI PyCharm 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: Setup Java |  | ||||||
|         uses: actions/setup-java@v4 |  | ||||||
|         with: |  | ||||||
|           distribution: zulu |  | ||||||
|           java-version: 17 |  | ||||||
|       - uses: actions/setup-python@v5 |  | ||||||
|         with: |  | ||||||
|           python-version: '3.10' |  | ||||||
|       - 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 -PideaType=PC > 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 :tests:ui-py-tests:testUi |  | ||||||
|       - name: Move video |  | ||||||
|         if: always() |  | ||||||
|         run: mv tests/ui-py-tests/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 |  | ||||||
							
								
								
									
										33
									
								
								.github/workflows/runUiTests.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										33
									
								
								.github/workflows/runUiTests.yml
									
									
									
									
										vendored
									
									
								
							| @@ -5,51 +5,44 @@ on: | |||||||
|       - cron: '0 12 * * *' |       - cron: '0 12 * * *' | ||||||
| jobs: | jobs: | ||||||
|   build-for-ui-test-mac-os: |   build-for-ui-test-mac-os: | ||||||
|     if: github.repository == 'JetBrains/ideavim' |  | ||||||
|     runs-on: macos-latest |     runs-on: macos-latest | ||||||
|     steps: |     steps: | ||||||
|       - uses: actions/checkout@v4 |       - uses: actions/checkout@v2 | ||||||
|       - name: Setup Java |       - name: Setup Java | ||||||
|         uses: actions/setup-java@v4 |         uses: actions/setup-java@v2.1.0 | ||||||
|         with: |         with: | ||||||
|           distribution: zulu |           distribution: zulu | ||||||
|           java-version: 17 |           java-version: 11 | ||||||
|       - name: Setup FFmpeg |       - name: Setup FFmpeg | ||||||
|         uses: FedericoCarboni/setup-ffmpeg@v3 |         uses: FedericoCarboni/setup-ffmpeg@v1 | ||||||
|         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. | ||||||
|           github-token: ${{ secrets.GITHUB_TOKEN }} |           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 | ||||||
|         run: | |         run: | | ||||||
|           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@v3 |         uses: jtalk/url-health-check-action@1.5 | ||||||
|         with: |         with: | ||||||
|           url: http://127.0.0.1:8082 |           url: http://127.0.0.1:8082 | ||||||
|           max-attempts: 20 |           max-attempts: 20 | ||||||
|           retry-delay: 10s |           retry-delay: 10s | ||||||
|       - name: Tests |       - name: Tests | ||||||
|         run: gradle :tests:ui-ij-tests:testUi |         run: gradle :testUi | ||||||
|       - name: Move video |       - name: Move video | ||||||
|         if: always() |         if: ${{ failure() }} | ||||||
|         run: mv tests/ui-ij-tests/video build/reports |         run: mv video build/reports | ||||||
|       - name: Move sandbox logs |       - name: Save fails report | ||||||
|         if: always() |         if: ${{ failure() }} | ||||||
|         run: mv build/idea-sandbox/system/log sandbox-idea-log |         uses: actions/upload-artifact@v2 | ||||||
|       - 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
									
								
								.github/workflows/syncDoc.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								.github/workflows/syncDoc.yml
									
									
									
									
										vendored
									
									
								
							| @@ -14,7 +14,6 @@ jobs: | |||||||
|   build: |   build: | ||||||
|  |  | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     if: github.repository == 'JetBrains/ideavim' |  | ||||||
|  |  | ||||||
|     steps: |     steps: | ||||||
|       - name: Fetch origin repo |       - name: Fetch origin repo | ||||||
|   | |||||||
							
								
								
									
										1
									
								
								.github/workflows/updateAffectedRate.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								.github/workflows/updateAffectedRate.yml
									
									
									
									
										vendored
									
									
								
							| @@ -14,7 +14,6 @@ jobs: | |||||||
|   build: |   build: | ||||||
|  |  | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     if: github.repository == 'JetBrains/ideavim' |  | ||||||
|  |  | ||||||
|     steps: |     steps: | ||||||
|       - name: Fetch origin repo |       - name: Fetch origin repo | ||||||
|   | |||||||
							
								
								
									
										1
									
								
								.github/workflows/updateAuthors.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								.github/workflows/updateAuthors.yml
									
									
									
									
										vendored
									
									
								
							| @@ -15,7 +15,6 @@ jobs: | |||||||
|   build: |   build: | ||||||
|  |  | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     if: github.repository == 'JetBrains/ideavim' |  | ||||||
|  |  | ||||||
|     steps: |     steps: | ||||||
|       - uses: actions/checkout@v3 |       - uses: actions/checkout@v3 | ||||||
|   | |||||||
							
								
								
									
										1
									
								
								.github/workflows/updateChangelog.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								.github/workflows/updateChangelog.yml
									
									
									
									
										vendored
									
									
								
							| @@ -15,7 +15,6 @@ jobs: | |||||||
|   build: |   build: | ||||||
|  |  | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     if: github.repository == 'JetBrains/ideavim' |  | ||||||
|  |  | ||||||
|     steps: |     steps: | ||||||
|       - uses: actions/checkout@v3 |       - uses: actions/checkout@v3 | ||||||
|   | |||||||
							
								
								
									
										1
									
								
								.github/workflows/updateFormatting.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								.github/workflows/updateFormatting.yml
									
									
									
									
										vendored
									
									
								
							| @@ -12,7 +12,6 @@ jobs: | |||||||
|   build: |   build: | ||||||
|  |  | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     if: github.repository == 'JetBrains/ideavim' |  | ||||||
|  |  | ||||||
|     steps: |     steps: | ||||||
|       - uses: actions/checkout@v2 |       - uses: actions/checkout@v2 | ||||||
|   | |||||||
							
								
								
									
										4
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -23,10 +23,6 @@ | |||||||
|  |  | ||||||
| # 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 |  | ||||||
| /vim-engine/src/main/resources/ksp-generated |  | ||||||
| /src/main/resources/ksp-generated |  | ||||||
|  |  | ||||||
| # Created by github automation | # Created by github automation | ||||||
| settings.xml | settings.xml | ||||||
|   | |||||||
							
								
								
									
										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-&#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="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="myName" value="IdeaVim" /> |     <option name="myName" value="IdeaVim" /> | ||||||
|   </copyright> |   </copyright> | ||||||
| </component> | </component> | ||||||
							
								
								
									
										18
									
								
								.teamcity/_Self/Constants.kt
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										18
									
								
								.teamcity/_Self/Constants.kt
									
									
									
									
										vendored
									
									
								
							| @@ -5,13 +5,15 @@ 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.3.2" |   const val VERSION = "2.4.0" | ||||||
|   const val NVIM_TESTS = "2023.3.2" |  | ||||||
|   const val PROPERTY_TESTS = "2023.3.2" |  | ||||||
|   const val LONG_RUNNING_TESTS = "2023.3.2" |  | ||||||
|   const val QODANA_TESTS = "2023.3.2" |  | ||||||
|   const val RELEASE = "2023.3.2" |  | ||||||
|  |  | ||||||
|   const val RELEASE_DEV = "2023.3.2" |   const val GITHUB_TESTS = "LATEST-EAP-SNAPSHOT" | ||||||
|   const val RELEASE_EAP = "2023.3.2" |   const val NVIM_TESTS = "LATEST-EAP-SNAPSHOT" | ||||||
|  |   const val PROPERTY_TESTS = "LATEST-EAP-SNAPSHOT" | ||||||
|  |   const val LONG_RUNNING_TESTS = "LATEST-EAP-SNAPSHOT" | ||||||
|  |   const val QODANA_TESTS = "LATEST-EAP-SNAPSHOT" | ||||||
|  |   const val RELEASE = "2023.1.2" | ||||||
|  |  | ||||||
|  |   const val RELEASE_DEV = "2023.1.2" | ||||||
|  |   const val RELEASE_EAP = "2023.1.2" | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										8
									
								
								.teamcity/_Self/Project.kt
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										8
									
								
								.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 JetBrains IDEs" |   description = "Vim engine for IDEs based on the IntelliJ platform" | ||||||
|  |  | ||||||
|   subProjects(Releases, OldTests, GitHub) |   subProjects(Releases, OldTests, GitHub) | ||||||
|  |  | ||||||
| @@ -24,7 +24,6 @@ object Project : Project({ | |||||||
|  |  | ||||||
|   // Active tests |   // Active tests | ||||||
|   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) | ||||||
| @@ -39,11 +38,6 @@ 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 { | ||||||
|   | |||||||
							
								
								
									
										1
									
								
								.teamcity/_Self/buildTypes/Compatibility.kt
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								.teamcity/_Self/buildTypes/Compatibility.kt
									
									
									
									
										vendored
									
									
								
							| @@ -34,6 +34,7 @@ object Compatibility : IdeaVimBuildType({ | |||||||
|                |                | ||||||
|               java --version |               java --version | ||||||
|               java -jar verifier1/verifier-cli-dev-all-1.jar check-plugin '${'$'}org.jetbrains.IdeaVim-EasyMotion' [latest-IU] -team-city |               java -jar verifier1/verifier-cli-dev-all-1.jar check-plugin '${'$'}org.jetbrains.IdeaVim-EasyMotion' [latest-IU] -team-city | ||||||
|  |               java -jar verifier1/verifier-cli-dev-all-1.jar check-plugin '${'$'}io.github.mishkun.ideavimsneak' [latest-IU] -team-city | ||||||
|               java -jar verifier1/verifier-cli-dev-all-1.jar check-plugin '${'$'}eu.theblob42.idea.whichkey' [latest-IU] -team-city |               java -jar verifier1/verifier-cli-dev-all-1.jar check-plugin '${'$'}eu.theblob42.idea.whichkey' [latest-IU] -team-city | ||||||
|               java -jar verifier1/verifier-cli-dev-all-1.jar check-plugin '${'$'}IdeaVimExtension' [latest-IU] -team-city |               java -jar verifier1/verifier-cli-dev-all-1.jar check-plugin '${'$'}IdeaVimExtension' [latest-IU] -team-city | ||||||
|               # Outdated java -jar verifier/verifier-cli-dev-all.jar check-plugin '${'$'}github.zgqq.intellij-enhance' [latest-IU] -team-city |               # Outdated java -jar verifier/verifier-cli-dev-all.jar check-plugin '${'$'}github.zgqq.intellij-enhance' [latest-IU] -team-city | ||||||
|   | |||||||
							
								
								
									
										2
									
								
								.teamcity/_Self/buildTypes/LongRunning.kt
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.teamcity/_Self/buildTypes/LongRunning.kt
									
									
									
									
										vendored
									
									
								
							| @@ -25,7 +25,7 @@ object LongRunning : IdeaVimBuildType({ | |||||||
|  |  | ||||||
|   steps { |   steps { | ||||||
|     gradle { |     gradle { | ||||||
|       tasks = "clean :tests:long-running-tests:testLongRunning" |       tasks = "clean testLongRunning" | ||||||
|       buildFile = "" |       buildFile = "" | ||||||
|       enableStacktrace = true |       enableStacktrace = true | ||||||
|     } |     } | ||||||
|   | |||||||
							
								
								
									
										2
									
								
								.teamcity/_Self/buildTypes/Nvim.kt
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.teamcity/_Self/buildTypes/Nvim.kt
									
									
									
									
										vendored
									
									
								
							| @@ -39,7 +39,7 @@ object Nvim : IdeaVimBuildType({ | |||||||
|               """.trimIndent() |               """.trimIndent() | ||||||
|     } |     } | ||||||
|     gradle { |     gradle { | ||||||
|       tasks = "clean test -Dnvim" |       tasks = "clean testWithNeovim" | ||||||
|       buildFile = "" |       buildFile = "" | ||||||
|       enableStacktrace = true |       enableStacktrace = true | ||||||
|     } |     } | ||||||
|   | |||||||
							
								
								
									
										2
									
								
								.teamcity/_Self/buildTypes/PropertyBased.kt
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.teamcity/_Self/buildTypes/PropertyBased.kt
									
									
									
									
										vendored
									
									
								
							| @@ -24,7 +24,7 @@ object PropertyBased : IdeaVimBuildType({ | |||||||
|  |  | ||||||
|   steps { |   steps { | ||||||
|     gradle { |     gradle { | ||||||
|       tasks = "clean :tests:property-tests:testPropertyBased" |       tasks = "clean testPropertyBased" | ||||||
|       buildFile = "" |       buildFile = "" | ||||||
|       enableStacktrace = true |       enableStacktrace = true | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -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:5ea56f8c-efe7-4e1e-83de-0c02bcc39d0b", display = ParameterDisplay.HIDDEN) |     password("env.ORG_GRADLE_PROJECT_spacePassword", "credentialsJSON:790b4e43-ee83-4184-b81b-678afab60409", display = ParameterDisplay.HIDDEN) | ||||||
|     param("env.ORG_GRADLE_PROJECT_spaceUsername", "a121c67e-39ac-40e6-bf82-649855ec27b6") |     param("env.ORG_GRADLE_PROJECT_spaceUsername", "Aleksei.Plate") | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   vcs { |   vcs { | ||||||
|   | |||||||
							
								
								
									
										16
									
								
								.teamcity/_Self/buildTypes/ReleaseEap.kt
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										16
									
								
								.teamcity/_Self/buildTypes/ReleaseEap.kt
									
									
									
									
										vendored
									
									
								
							| @@ -5,7 +5,6 @@ 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 | ||||||
| @@ -31,11 +30,6 @@ 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 { | ||||||
| @@ -67,9 +61,13 @@ 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 = """ | ||||||
| @@ -83,10 +81,6 @@ 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 { | ||||||
|   | |||||||
							
								
								
									
										65
									
								
								.teamcity/_Self/buildTypes/ReleasePlugin.kt
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										65
									
								
								.teamcity/_Self/buildTypes/ReleasePlugin.kt
									
									
									
									
										vendored
									
									
								
							| @@ -45,11 +45,7 @@ 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( |     password("env.ORG_GRADLE_PROJECT_youtrackToken", "credentialsJSON:3cd3e867-282c-451f-b958-bc95d56a8450", display = ParameterDisplay.HIDDEN) | ||||||
|       "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) | ||||||
|   } |   } | ||||||
|  |  | ||||||
| @@ -69,25 +65,9 @@ sealed class ReleasePlugin(private val releaseType: String) : IdeaVimBuildType({ | |||||||
|       name = "Pull git history" |       name = "Pull git history" | ||||||
|       scriptContent = "git fetch --unshallow" |       scriptContent = "git fetch --unshallow" | ||||||
|     } |     } | ||||||
|     script { |     gradle { | ||||||
|       name = "Reset release branch" |       name = "Select branch" | ||||||
|       scriptContent = """ |       tasks = "scripts:selectBranch" | ||||||
|         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" | ||||||
| @@ -109,47 +89,40 @@ sealed class ReleasePlugin(private val releaseType: String) : IdeaVimBuildType({ | |||||||
|       name = "Add release tag" |       name = "Add release tag" | ||||||
|       tasks = "scripts:addReleaseTag" |       tasks = "scripts:addReleaseTag" | ||||||
|     } |     } | ||||||
|     script { |     gradle { | ||||||
|       name = "Run tests" |       name = "Reset release branch" | ||||||
|       scriptContent = "./gradlew test" |       tasks = "scripts:resetReleaseBranch" | ||||||
|     } |     } | ||||||
|     gradle { |     gradle { | ||||||
|       name = "Publish release" |       name = "Publish release" | ||||||
|       tasks = "publishPlugin" |       tasks = "publishPlugin" | ||||||
|     } |     } | ||||||
|     script { | //    gradle { | ||||||
|       name = "Checkout master branch" | //      name = "Push changes to the repo" | ||||||
|       scriptContent = """ | //      tasks = "scripts:pushChangesWithReleaseBranch" | ||||||
|         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  ]; | ||||||
|  |       then | ||||||
|         git checkout release |         git checkout release | ||||||
|         echo checkout release branch |         echo checkout release branch | ||||||
|       git branch --set-upstream-to=origin/release release |         git push --tags | ||||||
|         git push origin --force |         git push origin --force | ||||||
|       # Push tag |       fi | ||||||
|       git push origin %build.number% |        | ||||||
|  |       git checkout ${'$'}branch | ||||||
|       """.trimIndent() |       """.trimIndent() | ||||||
|     } |     } | ||||||
|     gradle { |     gradle { | ||||||
|   | |||||||
							
								
								
									
										2
									
								
								.teamcity/_Self/subprojects/OldTests.kt
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.teamcity/_Self/subprojects/OldTests.kt
									
									
									
									
										vendored
									
									
								
							| @@ -20,6 +20,4 @@ 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)) |  | ||||||
| }) | }) | ||||||
|   | |||||||
| @@ -1,29 +0,0 @@ | |||||||
| 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" |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,19 +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 = 'PublishVimEngine' |  | ||||||
| accordingly, and delete the patch script. |  | ||||||
| */ |  | ||||||
| changeBuildType(RelativeId("PublishVimEngine")) { |  | ||||||
|     vcs { |  | ||||||
|  |  | ||||||
|         check(branchFilter == "+:<default>") { |  | ||||||
|             "Unexpected option value: branchFilter = $branchFilter" |  | ||||||
|         } |  | ||||||
|         branchFilter = "+:fleet" |  | ||||||
|     } |  | ||||||
| } |  | ||||||
							
								
								
									
										20
									
								
								.teamcity/patches/buildTypes/ReleasePatch.kts
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										20
									
								
								.teamcity/patches/buildTypes/ReleasePatch.kts
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,20 @@ | |||||||
|  | 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.11" | version = "2023.05" | ||||||
| project(_Self.Project) | project(_Self.Project) | ||||||
|   | |||||||
							
								
								
									
										12
									
								
								AUTHORS.md
									
									
									
									
									
								
							
							
						
						
									
										12
									
								
								AUTHORS.md
									
									
									
									
									
								
							| @@ -483,18 +483,6 @@ Contributors: | |||||||
|   [![icon][github]](https://github.com/ludwig-jb) |   [![icon][github]](https://github.com/ludwig-jb) | ||||||
|     |     | ||||||
|   ludwig-jb |   ludwig-jb | ||||||
| * [![icon][mail]](mailto:pvydmuch@gmail.com) |  | ||||||
|   [![icon][github]](https://github.com/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: | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										75
									
								
								CHANGES.md
									
									
									
									
									
								
							
							
						
						
									
										75
									
								
								CHANGES.md
									
									
									
									
									
								
							| @@ -25,86 +25,11 @@ 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 |  | ||||||
| * [VIM-3291](https://youtrack.jetbrains.com/issue/VIM-3291) Remove sync of editor selection between different opened editors |  | ||||||
| * [VIM-3234](https://youtrack.jetbrains.com/issue/VIM-3234) The space character won't mix in the tab chars after >> and << commands |  | ||||||
|  |  | ||||||
| ### Merged PRs: |  | ||||||
| * [725](https://github.com/JetBrains/ideavim/pull/725) by [Emanuel Gestosa](https://github.com/emanuelgestosa): Regex |  | ||||||
| * [805](https://github.com/JetBrains/ideavim/pull/805) by [chylex](https://github.com/chylex): VIM-3238 Fix recording a macro that replays another macro |  | ||||||
|  |  | ||||||
| ## 2.8.0, 2024-01-30 |  | ||||||
|  |  | ||||||
| ### Fixes: |  | ||||||
| * [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-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-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 |  | ||||||
|  |  | ||||||
| ### Fixes: |  | ||||||
| * [VIM-2933](https://youtrack.jetbrains.com/issue/VIM-2933) Reloading/sourcing .ideavimrc does not initialize new plugins |  | ||||||
| * [VIM-3138](https://youtrack.jetbrains.com/issue/VIM-3138) Do not try to register disposer if the caret is already disposed |  | ||||||
|  |  | ||||||
| ### Merged PRs: |  | ||||||
| * [734](https://github.com/JetBrains/ideavim/pull/734) by [Matt Ellis](https://github.com/citizenmatt): Support `~/` on Windows |  | ||||||
| * [736](https://github.com/JetBrains/ideavim/pull/736) by [chylex](https://github.com/chylex): Fix(VIM-2933): Reloading/sourcing .ideavimrc does not initialize new plugins |  | ||||||
|  |  | ||||||
| ## 2.6.3, 2023-10-30 |  | ||||||
|  |  | ||||||
| ### Changes: |  | ||||||
| - 2.6.0 and 2.6.1 releases are broken. Version 2.6.3 reverts IdeaVim plugin to the working state as for 2.5.1. |  | ||||||
|  |  | ||||||
| ## 2.6.0, 2023-10-27 |  | ||||||
|  |  | ||||||
| This version of IdeaVim contains a lot of issues. Version 2.6.3 reverts these changes. |  | ||||||
|  |  | ||||||
| ### Features: |  | ||||||
|  |  | ||||||
| * `ShowHoverInfo` action can be used in mappings to open a tooltip that is shown by |  | ||||||
|   mouse hovering | [VIM-2106](https://youtrack.jetbrains.com/issue/VIM-2106) |  | ||||||
| * `has` Vim Script function supports the most common OS checks: win32, win64, linux, mac, macunix, osx, osxdarwin, bsd, sun, unix |  | ||||||
|   * See https://github.com/JetBrains/ideavim#vim-script for details about Vim Script |  | ||||||
|  |  | ||||||
| ### Fixes: | ### Fixes: | ||||||
| * [VIM-3060](https://youtrack.jetbrains.com/issue/VIM-3060) Clipboard interaction stopped working | * [VIM-3060](https://youtrack.jetbrains.com/issue/VIM-3060) Clipboard interaction stopped working | ||||||
| * [VIM-3095](https://youtrack.jetbrains.com/issue/VIM-3095) Fix missing ellipsis digraph |  | ||||||
| * [VIM-2562](https://youtrack.jetbrains.com/issue/VIM-2562) Fix hang with multi-width chars in command line |  | ||||||
| * [VIM-696](https://youtrack.jetbrains.com/issue/VIM-696) Vim selection issue after undo |  | ||||||
| * [VIM-1639](https://youtrack.jetbrains.com/issue/VIM-1639) Ctrl-o and Ctrl-i jumping in files of different projects |  | ||||||
|  |  | ||||||
| ### Merged PRs: | ### Merged PRs: | ||||||
| * [697](https://github.com/JetBrains/ideavim/pull/697) by [Matt Ellis](https://github.com/citizenmatt): Support per-window "global" values for local-to-window options | * [697](https://github.com/JetBrains/ideavim/pull/697) by [Matt Ellis](https://github.com/citizenmatt): Support per-window "global" values for local-to-window options | ||||||
| * [717](https://github.com/JetBrains/ideavim/pull/717) by [Matt Ellis](https://github.com/citizenmatt): Fix(VIM-2562): Fix hang with multi-width chars in command line |  | ||||||
| * [732](https://github.com/JetBrains/ideavim/pull/732) by [pWydmuch](https://github.com/pWydmuch): Fix md links in doc |  | ||||||
| * [733](https://github.com/JetBrains/ideavim/pull/733) by [Matt Ellis](https://github.com/citizenmatt): Add support for ShowHoverInfo action to 2023.1 and 2023.2 |  | ||||||
| * [729](https://github.com/JetBrains/ideavim/pull/729) by [chylex](https://github.com/chylex): Add operating system type to `has()` function |  | ||||||
| * [726](https://github.com/JetBrains/ideavim/pull/726) by [Matt Ellis](https://github.com/citizenmatt): Fix range for fall back comment mode |  | ||||||
|  |  | ||||||
| ## 2.5.0, 2023-09-01 | ## 2.5.0, 2023-09-01 | ||||||
|  |  | ||||||
|   | |||||||
| @@ -255,7 +255,8 @@ Ex commands or via `:map` command mappings: | |||||||
| ##### Some popular actions: | ##### Some popular actions: | ||||||
|  |  | ||||||
| ``` | ``` | ||||||
| ShowHoverInfo - Quick Documentation and Error Description | QuickJavaDoc - Quick Documentation (not only for java, all languages) | ||||||
|  | ShowErrorDescription - Show description of the error under the caret (cursor hovering) | ||||||
| QuickImplementations - Quick Definition | QuickImplementations - Quick Definition | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
| @@ -324,7 +325,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 IdeaVim global marks: `set ideamarks` (works for marks with capital letters only) |     - Sync IJ bookmarks and Vim marks: `set ideamarks` | ||||||
|     - 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,8 +84,3 @@ 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,11 +8,8 @@ | |||||||
|  |  | ||||||
| plugins { | plugins { | ||||||
|   kotlin("jvm") |   kotlin("jvm") | ||||||
|   kotlin("plugin.serialization") version "1.9.22" |  | ||||||
| } | } | ||||||
|  |  | ||||||
| val kotlinxSerializationVersion: String by project |  | ||||||
|  |  | ||||||
| group = "com.intellij" | group = "com.intellij" | ||||||
| version = "SNAPSHOT" | version = "SNAPSHOT" | ||||||
|  |  | ||||||
| @@ -21,10 +18,6 @@ repositories { | |||||||
| } | } | ||||||
|  |  | ||||||
| dependencies { | dependencies { | ||||||
|   compileOnly("com.google.devtools.ksp:symbol-processing-api:1.9.22-1.0.17") |   compileOnly("com.google.devtools.ksp:symbol-processing-api:1.9.10-1.0.13") | ||||||
|   implementation("org.jetbrains.kotlinx:kotlinx-serialization-json-jvm:$kotlinxSerializationVersion") { |   implementation("org.jetbrains.kotlinx:kotlinx-serialization-json-jvm:1.6.0") | ||||||
|     // 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") |  | ||||||
|   } |  | ||||||
| } | } | ||||||
| @@ -1,55 +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.intellij.vim.annotations |  | ||||||
|  |  | ||||||
| // TODO support numpad keys parsing, see :keycodes |  | ||||||
| /** |  | ||||||
|  * It's not necessary a Vim command |  | ||||||
|  * This annotation may be used for: |  | ||||||
|  * - commands |  | ||||||
|  * - motions |  | ||||||
|  */ |  | ||||||
| @Target(AnnotationTarget.CLASS) |  | ||||||
| @Retention(AnnotationRetention.SOURCE) |  | ||||||
| annotation class CommandOrMotion(val keys: Array<String>, vararg val modes: Mode) |  | ||||||
|  |  | ||||||
| annotation class TextObject(val keys: String) |  | ||||||
|  |  | ||||||
|  |  | ||||||
| enum class Mode(val abbrev: Char) { |  | ||||||
|   /** |  | ||||||
|    * Indicates this key mapping applies to Normal mode |  | ||||||
|    */ |  | ||||||
|   NORMAL('N'), |  | ||||||
|  |  | ||||||
|   /** |  | ||||||
|    * Indicates this key mapping applies to Visual mode |  | ||||||
|    */ |  | ||||||
|   VISUAL('X'), |  | ||||||
|  |  | ||||||
|   /** |  | ||||||
|    * Indicates this key mapping applies to Select mode |  | ||||||
|    */ |  | ||||||
|   SELECT('S'), |  | ||||||
|  |  | ||||||
|   /** |  | ||||||
|    * Indicates this key mapping applies to Operator Pending mode |  | ||||||
|    */ |  | ||||||
|   OP_PENDING('O'), |  | ||||||
|  |  | ||||||
|   /** |  | ||||||
|    * Indicates this key mapping applies to Insert or Replace modes |  | ||||||
|    */ |  | ||||||
|   INSERT('I'), |  | ||||||
|  |  | ||||||
|   /** |  | ||||||
|    * Indicates this key mapping applies to Command Line mode |  | ||||||
|    */ |  | ||||||
|   CMD_LINE('C'), |  | ||||||
| } |  | ||||||
| @@ -1,62 +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.intellij.vim.processors |  | ||||||
|  |  | ||||||
| import com.google.devtools.ksp.KspExperimental |  | ||||||
| import com.google.devtools.ksp.getAnnotationsByType |  | ||||||
| import com.google.devtools.ksp.processing.Resolver |  | ||||||
| import com.google.devtools.ksp.processing.SymbolProcessor |  | ||||||
| import com.google.devtools.ksp.processing.SymbolProcessorEnvironment |  | ||||||
| import com.google.devtools.ksp.symbol.KSAnnotated |  | ||||||
| import com.google.devtools.ksp.symbol.KSClassDeclaration |  | ||||||
| import com.google.devtools.ksp.symbol.KSFile |  | ||||||
| import com.google.devtools.ksp.symbol.KSVisitorVoid |  | ||||||
| import com.intellij.vim.annotations.CommandOrMotion |  | ||||||
| import kotlinx.serialization.encodeToString |  | ||||||
| import kotlinx.serialization.json.Json |  | ||||||
| import java.nio.file.Files |  | ||||||
| import kotlin.io.path.Path |  | ||||||
| import kotlin.io.path.writeText |  | ||||||
|  |  | ||||||
| class CommandOrMotionProcessor(private val environment: SymbolProcessorEnvironment): SymbolProcessor { |  | ||||||
|   private val visitor = CommandOrMotionVisitor() |  | ||||||
|   private val commands = mutableListOf<CommandBean>() |  | ||||||
|  |  | ||||||
|   private val json = Json { prettyPrint = true } |  | ||||||
|  |  | ||||||
|   override fun process(resolver: Resolver): List<KSAnnotated> { |  | ||||||
|     resolver.getAllFiles().forEach { it.accept(visitor, Unit) } |  | ||||||
|  |  | ||||||
|     val generatedDirPath = Path(environment.options["generated_directory"]!!) |  | ||||||
|     Files.createDirectories(generatedDirPath) |  | ||||||
|  |  | ||||||
|     val filePath = generatedDirPath.resolve(environment.options["commands_file"]!!) |  | ||||||
|     val fileContent = json.encodeToString(commands) |  | ||||||
|     filePath.writeText(fileContent) |  | ||||||
|  |  | ||||||
|     return emptyList() |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   private inner class CommandOrMotionVisitor : KSVisitorVoid() { |  | ||||||
|     @OptIn(KspExperimental::class) |  | ||||||
|     override fun visitClassDeclaration(classDeclaration: KSClassDeclaration, data: Unit) { |  | ||||||
|       val commandAnnotation = classDeclaration.getAnnotationsByType(CommandOrMotion::class).firstOrNull() ?: return |  | ||||||
|       for (key in commandAnnotation.keys) { |  | ||||||
|         commands.add( |  | ||||||
|           CommandBean(key, classDeclaration.qualifiedName!!.asString(), commandAnnotation.modes.map { it.abbrev }.joinToString(separator = "")) |  | ||||||
|         ) |  | ||||||
|       } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     override fun visitFile(file: KSFile, data: Unit) { |  | ||||||
|       file.declarations.forEach { it.accept(this, Unit) } |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
|  |  | ||||||
| } |  | ||||||
| @@ -20,7 +20,6 @@ import com.google.devtools.ksp.symbol.KSVisitorVoid | |||||||
| import com.intellij.vim.annotations.ExCommand | import com.intellij.vim.annotations.ExCommand | ||||||
| import kotlinx.serialization.encodeToString | import kotlinx.serialization.encodeToString | ||||||
| import kotlinx.serialization.json.Json | import kotlinx.serialization.json.Json | ||||||
| import java.nio.file.Files |  | ||||||
| import kotlin.io.path.Path | import kotlin.io.path.Path | ||||||
| import kotlin.io.path.writeText | import kotlin.io.path.writeText | ||||||
|  |  | ||||||
| @@ -32,11 +31,7 @@ class ExCommandProcessor(private val environment: SymbolProcessorEnvironment): S | |||||||
|  |  | ||||||
|   override fun process(resolver: Resolver): List<KSAnnotated> { |   override fun process(resolver: Resolver): List<KSAnnotated> { | ||||||
|     resolver.getAllFiles().forEach { it.accept(visitor, Unit) } |     resolver.getAllFiles().forEach { it.accept(visitor, Unit) } | ||||||
|  |     val filePath = Path(environment.options["generated_directory"]!!, environment.options["ex_commands_file"]!!) | ||||||
|     val generatedDirPath = Path(environment.options["generated_directory"]!!) |  | ||||||
|     Files.createDirectories(generatedDirPath) |  | ||||||
|  |  | ||||||
|     val filePath = generatedDirPath.resolve(environment.options["ex_commands_file"]!!) |  | ||||||
|     val fileContent = json.encodeToString(commandToClass) |     val fileContent = json.encodeToString(commandToClass) | ||||||
|     filePath.writeText(fileContent) |     filePath.writeText(fileContent) | ||||||
|  |  | ||||||
|   | |||||||
| @@ -20,7 +20,6 @@ import com.google.devtools.ksp.symbol.KSVisitorVoid | |||||||
| import com.intellij.vim.annotations.VimscriptFunction | import com.intellij.vim.annotations.VimscriptFunction | ||||||
| import kotlinx.serialization.encodeToString | import kotlinx.serialization.encodeToString | ||||||
| import kotlinx.serialization.json.Json | import kotlinx.serialization.json.Json | ||||||
| import java.nio.file.Files |  | ||||||
| import kotlin.io.path.Path | import kotlin.io.path.Path | ||||||
| import kotlin.io.path.writeText | import kotlin.io.path.writeText | ||||||
|  |  | ||||||
| @@ -32,11 +31,7 @@ class VimscriptFunctionProcessor(private val environment: SymbolProcessorEnviron | |||||||
|  |  | ||||||
|   override fun process(resolver: Resolver): List<KSAnnotated> { |   override fun process(resolver: Resolver): List<KSAnnotated> { | ||||||
|     resolver.getAllFiles().forEach { it.accept(visitor, Unit) } |     resolver.getAllFiles().forEach { it.accept(visitor, Unit) } | ||||||
|  |     val filePath = Path(environment.options["generated_directory"]!!, environment.options["vimscript_functions_file"]!!) | ||||||
|     val generatedDirPath = Path(environment.options["generated_directory"]!!) |  | ||||||
|     Files.createDirectories(generatedDirPath) |  | ||||||
|  |  | ||||||
|     val filePath = generatedDirPath.resolve(environment.options["vimscript_functions_file"]!!) |  | ||||||
|     val fileContent = json.encodeToString(nameToClass) |     val fileContent = json.encodeToString(nameToClass) | ||||||
|     filePath.writeText(fileContent) |     filePath.writeText(fileContent) | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,20 +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.intellij.vim.providers |  | ||||||
|  |  | ||||||
| import com.google.devtools.ksp.processing.SymbolProcessor |  | ||||||
| import com.google.devtools.ksp.processing.SymbolProcessorEnvironment |  | ||||||
| import com.google.devtools.ksp.processing.SymbolProcessorProvider |  | ||||||
| import com.intellij.vim.processors.CommandOrMotionProcessor |  | ||||||
|  |  | ||||||
| class CommandOrMotionProcessorProvider : SymbolProcessorProvider { |  | ||||||
|   override fun create(environment: SymbolProcessorEnvironment): SymbolProcessor { |  | ||||||
|     return CommandOrMotionProcessor(environment) |  | ||||||
|   } |  | ||||||
| } |  | ||||||
| @@ -1,3 +1,2 @@ | |||||||
| com.intellij.vim.providers.CommandOrMotionProcessorProvider |  | ||||||
| com.intellij.vim.providers.ExCommandProcessorProvider |  | ||||||
| com.intellij.vim.providers.VimscriptFunctionProcessorProvider | com.intellij.vim.providers.VimscriptFunctionProcessorProvider | ||||||
|  | com.intellij.vim.providers.ExCommandProcessorProvider | ||||||
							
								
								
									
										228
									
								
								build.gradle.kts
									
									
									
									
									
								
							
							
						
						
									
										228
									
								
								build.gradle.kts
									
									
									
									
									
								
							| @@ -44,19 +44,19 @@ buildscript { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     dependencies { |     dependencies { | ||||||
|     classpath("org.jetbrains.kotlin:kotlin-gradle-plugin:1.9.22") |         classpath("org.jetbrains.kotlin:kotlin-gradle-plugin:1.8.21") | ||||||
|         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.8.0.202311291450-r") |         classpath("org.eclipse.jgit:org.eclipse.jgit.ssh.apache:6.7.0.202309050840-r") | ||||||
|         classpath("org.kohsuke:github-api:1.305") |         classpath("org.kohsuke:github-api:1.305") | ||||||
|  |  | ||||||
|     classpath("io.ktor:ktor-client-core:2.3.7") |         classpath("io.ktor:ktor-client-core:2.3.4") | ||||||
|     classpath("io.ktor:ktor-client-cio:2.3.7") |         classpath("io.ktor:ktor-client-cio:2.3.4") | ||||||
|     classpath("io.ktor:ktor-client-auth:2.3.7") |         classpath("io.ktor:ktor-client-auth:2.3.4") | ||||||
|     classpath("io.ktor:ktor-client-content-negotiation:2.3.7") |         classpath("io.ktor:ktor-client-content-negotiation:2.3.4") | ||||||
|     classpath("io.ktor:ktor-serialization-kotlinx-json:2.3.7") |         classpath("io.ktor:ktor-serialization-kotlinx-json:2.3.4") | ||||||
|  |  | ||||||
|         // 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,31 +66,30 @@ buildscript { | |||||||
| plugins { | plugins { | ||||||
|     antlr |     antlr | ||||||
|     java |     java | ||||||
|   kotlin("jvm") version "1.9.22" |     kotlin("jvm") version "1.8.21" | ||||||
|     application |     application | ||||||
|   id("java-test-fixtures") |  | ||||||
|  |  | ||||||
|   id("org.jetbrains.intellij") version "1.17.1" |     id("org.jetbrains.intellij") version "1.15.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 | ||||||
|  | //    id("org.jlleitschuh.gradle.ktlint") version "11.3.1" | ||||||
|  |  | ||||||
|     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.9.22-1.0.17" |     id("com.google.devtools.ksp") version "1.8.21-1.0.11" | ||||||
| } | } | ||||||
|  |  | ||||||
| ksp { | ksp { | ||||||
|   arg("generated_directory", "$projectDir/src/main/resources/ksp-generated") |   arg("generated_directory", "$projectDir/src/main/resources") | ||||||
|   arg("vimscript_functions_file", "intellij_vimscript_functions.json") |   arg("vimscript_functions_file", "intellij_vimscript_functions.json") | ||||||
|   arg("ex_commands_file", "intellij_ex_commands.json") |   arg("ex_commands_file", "intellij_ex_commands.json") | ||||||
|   arg("commands_file", "intellij_commands.json") |  | ||||||
| } | } | ||||||
|  |  | ||||||
| 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,11 +97,10 @@ 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 antlrVersion: String by project |  | ||||||
| val remoteRobotVersion: String by project | val remoteRobotVersion: String by project | ||||||
|  | val antlrVersion: String by project | ||||||
|  |  | ||||||
| val publishChannels: String by project | val publishChannels: String by project | ||||||
| val publishToken: String by project | val publishToken: String by project | ||||||
| @@ -116,40 +114,35 @@ 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.1.0") |     compileOnly("org.jetbrains:annotations:24.0.1") | ||||||
|   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.2.1") |     testImplementation("org.mockito.kotlin:mockito-kotlin:5.1.0") | ||||||
|  |  | ||||||
|   testImplementation("org.junit.jupiter:junit-jupiter-api:5.10.1") |     testImplementation("com.intellij.remoterobot:remote-robot:$remoteRobotVersion") | ||||||
|   testImplementation("org.junit.jupiter:junit-jupiter-engine:5.10.1") |     testImplementation("com.intellij.remoterobot:remote-fixtures:$remoteRobotVersion") | ||||||
|   testImplementation("org.junit.jupiter:junit-jupiter-params:5.10.1") |     testImplementation("com.automation-remarks:video-recorder-junit:2.0") | ||||||
|   testFixturesImplementation("org.junit.jupiter:junit-jupiter-api:5.10.1") |     runtimeOnly("org.antlr:antlr4-runtime:$antlrVersion") | ||||||
|   testFixturesImplementation("org.junit.jupiter:junit-jupiter-engine:5.10.1") |     antlr("org.antlr:antlr4:$antlrVersion") | ||||||
|   testFixturesImplementation("org.junit.jupiter:junit-jupiter-params:5.10.1") |  | ||||||
|  |     api(project(":vim-engine")) | ||||||
|  |  | ||||||
|  |     ksp(project(":annotation-processors")) | ||||||
|  |     compileOnly(project(":annotation-processors")) | ||||||
|  |  | ||||||
|  |     testApi("com.squareup.okhttp3:okhttp:4.11.0") | ||||||
|  |  | ||||||
|  |     testImplementation("org.junit.jupiter:junit-jupiter-api:5.10.0") | ||||||
|  |     testImplementation("org.junit.jupiter:junit-jupiter-engine:5.10.0") | ||||||
|  |     testImplementation("org.junit.jupiter:junit-jupiter-params:5.10.0") | ||||||
| } | } | ||||||
|  |  | ||||||
| configurations { | configurations { | ||||||
| @@ -158,17 +151,66 @@ configurations { | |||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
| tasks { | // --- Compilation | ||||||
|   test { | // This can be moved to other test registration when issue with tests in gradle will be fixed | ||||||
|     // Set teamcity env variable locally to run additional tests for leaks. | tasks.register<Test>("testWithNeovim") { | ||||||
|     // By default, this test runs on TC only, but this test doesn't take a lot of time, |     group = "verification" | ||||||
|     //   so we can turn it on for local development |     systemProperty("ideavim.nvim.test", "true") | ||||||
|     if (environment["TEAMCITY_VERSION"] == null) { |     exclude("/ui/**") | ||||||
|       println("Set env TEAMCITY_VERSION to X to enable project leak checks from the platform") |     exclude("**/longrunning/**") | ||||||
|       environment("TEAMCITY_VERSION" to "X") |     exclude("**/propertybased/**") | ||||||
|  |     useJUnitPlatform() | ||||||
| } | } | ||||||
|  |  | ||||||
|     systemProperty("ideavim.nvim.test", System.getProperty("nvim") ?: false) | tasks.register<Test>("testPropertyBased") { | ||||||
|  |     group = "verification" | ||||||
|  | //    include("**/propertybased/**") | ||||||
|  |     useJUnitPlatform() | ||||||
|  | } | ||||||
|  |  | ||||||
|  | tasks.register<Test>("testLongRunning") { | ||||||
|  |     group = "verification" | ||||||
|  | //    include("**/longrunning/**") | ||||||
|  |     useJUnitPlatform() | ||||||
|  | } | ||||||
|  |  | ||||||
|  | tasks { | ||||||
|  |     // Issue in gradle 7.3 | ||||||
|  |     val test by getting(Test::class) { | ||||||
|  |         isScanForTestClasses = false | ||||||
|  |         // Only run tests from classes that end with "Test" | ||||||
|  |         include("**/*Test.class") | ||||||
|  |         include("**/*test.class") | ||||||
|  |         include("**/*Tests.class") | ||||||
|  |         exclude("**/ParserTest.class") | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     val testWithNeovim by getting(Test::class) { | ||||||
|  |         isScanForTestClasses = false | ||||||
|  |         // Only run tests from classes that end with "Test" | ||||||
|  |         include("**/*Test.class") | ||||||
|  |         include("**/*test.class") | ||||||
|  |         include("**/*Tests.class") | ||||||
|  |         exclude("**/ParserTest.class") | ||||||
|  |         exclude("**/longrunning/**") | ||||||
|  |         exclude("**/propertybased/**") | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     val testPropertyBased by getting(Test::class) { | ||||||
|  |         isScanForTestClasses = false | ||||||
|  |         // Only run tests from classes that end with "Test" | ||||||
|  |         include("**/propertybased/*Test.class") | ||||||
|  |         include("**/propertybased/*test.class") | ||||||
|  |         include("**/propertybased/*Tests.class") | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     val testLongRunning by getting(Test::class) { | ||||||
|  |         isScanForTestClasses = false | ||||||
|  |         // Only run tests from classes that end with "Test" | ||||||
|  |         include("**/longrunning/**/*Test.class") | ||||||
|  |         include("**/longrunning/**/*test.class") | ||||||
|  |         include("**/longrunning/**/*Tests.class") | ||||||
|  |         exclude("**/longrunning/**/ParserTest.class") | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     compileJava { |     compileJava { | ||||||
| @@ -181,33 +223,18 @@ tasks { | |||||||
|     compileKotlin { |     compileKotlin { | ||||||
|         kotlinOptions { |         kotlinOptions { | ||||||
|             jvmTarget = javaVersion |             jvmTarget = javaVersion | ||||||
|       // See https://plugins.jetbrains.com/docs/intellij/using-kotlin.html#kotlin-standard-library |             apiVersion = "1.6" | ||||||
|       // 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 | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     compileTestKotlin { |     compileTestKotlin { | ||||||
|         kotlinOptions { |         kotlinOptions { | ||||||
|             jvmTarget = javaVersion |             jvmTarget = javaVersion | ||||||
|       apiVersion = "1.9" |             apiVersion = "1.6" | ||||||
| //            allWarningsAsErrors = true | //            allWarningsAsErrors = true | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|   downloadRobotServerPlugin { |  | ||||||
|     version.set(remoteRobotVersion) |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   runIdeForUiTests { |  | ||||||
|     systemProperty("robot-server.port", "8082") |  | ||||||
|     systemProperty("ide.mac.message.dialogs.as.sheets", "false") |  | ||||||
|     systemProperty("jb.privacy.policy.text", "<!--999.999-->") |  | ||||||
|     systemProperty("jb.consents.confirmation.enabled", "false") |  | ||||||
|     systemProperty("ide.show.tips.on.startup.default.value", "false") |  | ||||||
|   } |  | ||||||
| } | } | ||||||
|  |  | ||||||
| java { | java { | ||||||
| @@ -234,7 +261,6 @@ gradle.projectsEvaluated { | |||||||
|  |  | ||||||
| intellij { | intellij { | ||||||
|     version.set(ideaVersion) |     version.set(ideaVersion) | ||||||
|   type.set(ideaType) |  | ||||||
|     pluginName.set("IdeaVim") |     pluginName.set("IdeaVim") | ||||||
|  |  | ||||||
|     updateSinceUntilBuild.set(false) |     updateSinceUntilBuild.set(false) | ||||||
| @@ -242,10 +268,15 @@ 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") | ||||||
|   plugins.set(listOf("AceJump:3.8.11")) |     // Yaml is only used for testing. It's part of the IdeaIC distribution, but needs to be included as a reference | ||||||
|  |     plugins.set(listOf("java", "AceJump:3.8.11", "yaml"/*, "Pythonid:231.8109.2", "com.intellij.clion-swift:231.8109.4"*/)) | ||||||
| } | } | ||||||
|  |  | ||||||
| tasks { | tasks { | ||||||
|  |     downloadRobotServerPlugin { | ||||||
|  |         version.set(remoteRobotVersion) | ||||||
|  |     } | ||||||
|  |  | ||||||
|     publishPlugin { |     publishPlugin { | ||||||
|         channels.set(publishChannels.split(",")) |         channels.set(publishChannels.split(",")) | ||||||
|         token.set(publishToken) |         token.set(publishToken) | ||||||
| @@ -257,6 +288,13 @@ tasks { | |||||||
|         password.set(providers.environmentVariable("PRIVATE_KEY_PASSWORD")) |         password.set(providers.environmentVariable("PRIVATE_KEY_PASSWORD")) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     runIdeForUiTests { | ||||||
|  |         systemProperty("robot-server.port", "8082") | ||||||
|  |         systemProperty("ide.mac.message.dialogs.as.sheets", "false") | ||||||
|  |         systemProperty("jb.privacy.policy.text", "<!--999.999-->") | ||||||
|  |         systemProperty("jb.consents.confirmation.enabled", "false") | ||||||
|  |     } | ||||||
|  |  | ||||||
|     runPluginVerifier { |     runPluginVerifier { | ||||||
|         downloadDir.set("${project.buildDir}/pluginVerifier/ides") |         downloadDir.set("${project.buildDir}/pluginVerifier/ides") | ||||||
|         teamCityOutputFormat.set(true) |         teamCityOutputFormat.set(true) | ||||||
| @@ -275,9 +313,6 @@ 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) { | ||||||
| @@ -308,6 +343,8 @@ 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") | ||||||
|  |  | ||||||
|         // Get the latest available change notes from the changelog file |         // Get the latest available change notes from the changelog file | ||||||
|         changeNotes.set( |         changeNotes.set( | ||||||
|             provider { |             provider { | ||||||
| @@ -327,14 +364,29 @@ tasks { | |||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | // --- Linting | ||||||
|  |  | ||||||
|  | //ktlint { | ||||||
|  | //    version.set("0.48.2") | ||||||
|  | //} | ||||||
|  |  | ||||||
| // --- Tests | // --- Tests | ||||||
|  |  | ||||||
| tasks { | tasks { | ||||||
|     test { |     test { | ||||||
|         useJUnitPlatform() |         useJUnitPlatform() | ||||||
|  |         exclude("**/propertybased/**") | ||||||
|  |         exclude("**/longrunning/**") | ||||||
|  |         exclude("/ui/**") | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | tasks.register<Test>("testUi") { | ||||||
|  |     group = "verification" | ||||||
|  |     useJUnitPlatform() | ||||||
|  |     include("/ui/**") | ||||||
|  | } | ||||||
|  |  | ||||||
| // --- Changelog | // --- Changelog | ||||||
|  |  | ||||||
| changelog { | changelog { | ||||||
| @@ -353,11 +405,20 @@ koverMerged { | |||||||
|     enable() |     enable() | ||||||
| } | } | ||||||
|  |  | ||||||
|  | kover { | ||||||
|  |     instrumentation { | ||||||
|  |         // set of test tasks names to exclude from instrumentation. The results of their execution will not be presented in the report | ||||||
|  |         excludeTasks += "testPropertyBased" | ||||||
|  |         excludeTasks += "testLongRunning" | ||||||
|  |         excludeTasks += "testWithNeovim" | ||||||
|  |         excludeTasks += "testUi" | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
| // --- Slack notification | // --- Slack notification | ||||||
|  |  | ||||||
| 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 | ||||||
| @@ -458,16 +519,14 @@ val fixVersionsElementType = "VersionBundleElement" | |||||||
| tasks.register("releaseActions") { | tasks.register("releaseActions") { | ||||||
|     group = "other" |     group = "other" | ||||||
|     doLast { |     doLast { | ||||||
|     val tickets = getYoutrackTicketsByQuery("%23%7BReady+To+Release%7D%20and%20tag:%20%7BIdeaVim%20Released%20In%20EAP%7D%20") |         val tickets = getYoutrackTicketsByQuery("%23%7BReady+To+Release%7D") | ||||||
|         if (tickets.isNotEmpty()) { |         if (tickets.isNotEmpty()) { | ||||||
|             println("Updating statuses for tickets: $tickets") |             println("Updating statuses for tickets: $tickets") | ||||||
|             setYoutrackStatus(tickets, "Fixed") |             setYoutrackStatus(tickets, "Fixed") | ||||||
|       println("Checking if version $version exists...") |             if (getVersionIdByName(version.toString()) != null) { | ||||||
|       val versionId = getVersionIdByName(version.toString()) |  | ||||||
|       if (versionId == null) { |  | ||||||
|                 addReleaseToYoutrack(version.toString()) |                 addReleaseToYoutrack(version.toString()) | ||||||
|             } else { |             } else { | ||||||
|         println("Version $version already exists in YouTrack. Version id: $versionId") |                 println("Version $version is already exists in YouTrack") | ||||||
|             } |             } | ||||||
|             setYoutrackFixVersion(tickets, version.toString()) |             setYoutrackFixVersion(tickets, version.toString()) | ||||||
|         } else { |         } else { | ||||||
| @@ -546,8 +605,7 @@ fun addReleaseToYoutrack(name: String): String { | |||||||
|     println("Creating new release version in YouTrack: $name") |     println("Creating new release version in YouTrack: $name") | ||||||
|  |  | ||||||
|     return runBlocking { |     return runBlocking { | ||||||
|     val response = |         val response = client.post("https://youtrack.jetbrains.com/api/admin/projects/$vimProjectId/customFields/$fixVersionsFieldId/bundle/values?fields=id,name") { | ||||||
|       client.post("https://youtrack.jetbrains.com/api/admin/projects/$vimProjectId/customFields/$fixVersionsFieldId/bundle/values?fields=id,name") { |  | ||||||
|             contentType(ContentType.Application.Json) |             contentType(ContentType.Application.Json) | ||||||
|             accept(ContentType.Application.Json) |             accept(ContentType.Application.Json) | ||||||
|             val request = buildJsonObject { |             val request = buildJsonObject { | ||||||
| @@ -564,8 +622,7 @@ fun getVersionIdByName(name: String): String? { | |||||||
|     val client = httpClient() |     val client = httpClient() | ||||||
|  |  | ||||||
|     return runBlocking { |     return runBlocking { | ||||||
|     val response = |         val response = client.get("https://youtrack.jetbrains.com/api/admin/projects/$vimProjectId/customFields/$fixVersionsFieldId/bundle/values?fields=id,name&query=$name") | ||||||
|       client.get("https://youtrack.jetbrains.com/api/admin/projects/$vimProjectId/customFields/$fixVersionsFieldId/bundle/values?fields=id,name&query=$name") |  | ||||||
|         response.body<JsonArray>().singleOrNull()?.jsonObject?.get("id")?.jsonPrimitive?.content |         response.body<JsonArray>().singleOrNull()?.jsonObject?.get("id")?.jsonPrimitive?.content | ||||||
|     } |     } | ||||||
| } | } | ||||||
| @@ -603,8 +660,7 @@ fun setYoutrackStatus(tickets: Collection<String>, status: String) { | |||||||
|     runBlocking { |     runBlocking { | ||||||
|         for (ticket in tickets) { |         for (ticket in tickets) { | ||||||
|             println("Try to set $ticket to $status") |             println("Try to set $ticket to $status") | ||||||
|       val response = |             val response = client.post("https://youtrack.jetbrains.com/api/issues/$ticket?fields=customFields(id,name,value(id,name))") { | ||||||
|         client.post("https://youtrack.jetbrains.com/api/issues/$ticket?fields=customFields(id,name,value(id,name))") { |  | ||||||
|                 contentType(ContentType.Application.Json) |                 contentType(ContentType.Application.Json) | ||||||
|                 accept(ContentType.Application.Json) |                 accept(ContentType.Application.Json) | ||||||
|                 val request = buildJsonObject { |                 val request = buildJsonObject { | ||||||
| @@ -643,8 +699,7 @@ fun setYoutrackFixVersion(tickets: Collection<String>, version: String) { | |||||||
|     runBlocking { |     runBlocking { | ||||||
|         for (ticket in tickets) { |         for (ticket in tickets) { | ||||||
|             println("Try to set fix version $version for $ticket") |             println("Try to set fix version $version for $ticket") | ||||||
|       val response = |             val response = client.post("https://youtrack.jetbrains.com/api/issues/$ticket?fields=customFields(id,name,value(id,name))") { | ||||||
|         client.post("https://youtrack.jetbrains.com/api/issues/$ticket?fields=customFields(id,name,value(id,name))") { |  | ||||||
|                 contentType(ContentType.Application.Json) |                 contentType(ContentType.Application.Json) | ||||||
|                 accept(ContentType.Application.Json) |                 accept(ContentType.Application.Json) | ||||||
|                 val request = buildJsonObject { |                 val request = buildJsonObject { | ||||||
| @@ -682,8 +737,7 @@ fun getYoutrackStatus(ticket: String): String { | |||||||
|     val client = httpClient() |     val client = httpClient() | ||||||
|  |  | ||||||
|     return runBlocking { |     return runBlocking { | ||||||
|     val response = |         val response = client.get("https://youtrack.jetbrains.com/api/issues/$ticket/customFields/123-129?fields=value(name)") | ||||||
|       client.get("https://youtrack.jetbrains.com/api/issues/$ticket/customFields/123-129?fields=value(name)") |  | ||||||
|         response.body<JsonObject>()["value"]!!.jsonObject.getValue("name").jsonPrimitive.content |         response.body<JsonObject>()["value"]!!.jsonObject.getValue("name").jsonPrimitive.content | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,6 +1,6 @@ | |||||||
| Welcome to the IdeaVim wiki! | Welcome to the IdeaVim wiki! | ||||||
|  |  | ||||||
| - List of IdeaVim plugins: [plugins](IdeaVim%20Plugins.md) | - List of IdeaVim plugins: [[plugins|IdeaVim Plugins]] | ||||||
| - Examples of `ideajoin` option (also known as "smart join"): ["ideajoin" examples](ideajoin-examples.md) | - Examples of `ideajoin` option (also known as "smart join"): [["ideajoin" examples|ideajoin-examples]] | ||||||
| - List of "set" commands: ["set" commands](set-commands.md) | - List of "set" commands: [["set" commands|set-commands]] | ||||||
| - Docs about "select" mode in vim: [select mode](Select-mode.md) | - Docs about "select" mode in vim: [[select mode|Select-mode]] | ||||||
|   | |||||||
| @@ -45,20 +45,15 @@ 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: | ||||||
| - Add the following command to `~/.ideavimrc`: `Plug 'justinmk/vim-sneak'` | - Install [IdeaVim-sneak](https://plugins.jetbrains.com/plugin/15348-ideavim-sneak) plugin. | ||||||
|  | - Add the following command to `~/.ideavimrc`: `set sneak` | ||||||
|     |     | ||||||
| ### Instructions | ### Instructions | ||||||
|     |     | ||||||
| * Type `s` and two chars to start sneaking in forward direction | See the [docs](https://github.com/Mishkun/ideavim-sneak#usage) | ||||||
| * 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> | ||||||
|  |  | ||||||
| @@ -82,7 +77,7 @@ Original plugin: [NERDTree](https://github.com/preservim/nerdtree). | |||||||
|     |     | ||||||
| ### Instructions | ### Instructions | ||||||
|     |     | ||||||
| [See here](NERDTree-support.md). | [[See here|NERDTree-support]]. | ||||||
|  |  | ||||||
| </details> | </details> | ||||||
|  |  | ||||||
| @@ -401,19 +396,3 @@ 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,11 +3,6 @@ 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: | ||||||
|  |  | ||||||
| ``` | ``` | ||||||
|   | |||||||
| @@ -1,28 +0,0 @@ | |||||||
| <?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> |  | ||||||
| Before Width: | Height: | Size: 5.6 KiB | 
| @@ -8,28 +8,21 @@ | |||||||
|  |  | ||||||
| # suppress inspection "UnusedProperty" for whole file | # suppress inspection "UnusedProperty" for whole file | ||||||
|  |  | ||||||
| ideaVersion=2023.3.3 | ideaVersion=2023.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=chylex-28 | version=chylex-18 | ||||||
| javaVersion=17 | javaVersion=17 | ||||||
| remoteRobotVersion=0.11.22 | remoteRobotVersion=0.11.17 | ||||||
| antlrVersion=4.10.1 | antlrVersion=4.10.1 | ||||||
|  |  | ||||||
| kotlin.incremental.useClasspathSnapshot=false | kotlin.incremental.useClasspathSnapshot=false | ||||||
|  |  | ||||||
| # Please don't forget to update kotlin version in buildscript section | # Please don't forget to update kotlin version in buildscript section | ||||||
| # Also update kotlinxSerializationVersion version | kotlinVersion=1.8.21 | ||||||
| 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= | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										
											BIN
										
									
								
								gradle/wrapper/gradle-wrapper.jar
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										
											BIN
										
									
								
								gradle/wrapper/gradle-wrapper.jar
									
									
									
									
										vendored
									
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										3
									
								
								gradle/wrapper/gradle-wrapper.properties
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										3
									
								
								gradle/wrapper/gradle-wrapper.properties
									
									
									
									
										vendored
									
									
								
							| @@ -1,7 +1,6 @@ | |||||||
| distributionBase=GRADLE_USER_HOME | distributionBase=GRADLE_USER_HOME | ||||||
| distributionPath=wrapper/dists | distributionPath=wrapper/dists | ||||||
| distributionUrl=https\://services.gradle.org/distributions/gradle-8.6-bin.zip | distributionUrl=https\://services.gradle.org/distributions/gradle-8.1.1-bin.zip | ||||||
| networkTimeout=10000 | networkTimeout=10000 | ||||||
| validateDistributionUrl=true |  | ||||||
| zipStoreBase=GRADLE_USER_HOME | zipStoreBase=GRADLE_USER_HOME | ||||||
| zipStorePath=wrapper/dists | zipStorePath=wrapper/dists | ||||||
|   | |||||||
							
								
								
									
										5
									
								
								gradlew
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										5
									
								
								gradlew
									
									
									
									
										vendored
									
									
								
							| @@ -130,14 +130,11 @@ location of your Java installation." | |||||||
|     fi |     fi | ||||||
| else | else | ||||||
|     JAVACMD=java |     JAVACMD=java | ||||||
|     if ! command -v java >/dev/null 2>&1 |     which java >/dev/null 2>&1 || die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. | ||||||
|     then |  | ||||||
|         die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. |  | ||||||
|  |  | ||||||
| Please set the JAVA_HOME variable in your environment to match the | Please set the JAVA_HOME variable in your environment to match the | ||||||
| location of your Java installation." | location of your Java installation." | ||||||
| fi | fi | ||||||
| fi |  | ||||||
|  |  | ||||||
| # Increase the maximum file descriptors if we can. | # Increase the maximum file descriptors if we can. | ||||||
| if ! "$cygwin" && ! "$darwin" && ! "$nonstop" ; then | if ! "$cygwin" && ! "$darwin" && ! "$nonstop" ; then | ||||||
|   | |||||||
| @@ -20,17 +20,17 @@ repositories { | |||||||
| } | } | ||||||
|  |  | ||||||
| dependencies { | dependencies { | ||||||
|   compileOnly("org.jetbrains.kotlin:kotlin-stdlib:1.9.22") |   compileOnly("org.jetbrains.kotlin:kotlin-stdlib:1.9.10") | ||||||
|  |  | ||||||
|   implementation("io.ktor:ktor-client-core:2.3.7") |   implementation("io.ktor:ktor-client-core:2.3.4") | ||||||
|   implementation("io.ktor:ktor-client-cio:2.3.7") |   implementation("io.ktor:ktor-client-cio:2.3.4") | ||||||
|   implementation("io.ktor:ktor-client-content-negotiation:2.3.7") |   implementation("io.ktor:ktor-client-content-negotiation:2.3.4") | ||||||
|   implementation("io.ktor:ktor-serialization-kotlinx-json:2.3.7") |   implementation("io.ktor:ktor-serialization-kotlinx-json:2.3.4") | ||||||
|   implementation("io.ktor:ktor-client-auth:2.3.7") |   implementation("io.ktor:ktor-client-auth:2.3.4") | ||||||
|   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.8.0.202311291450-r") |   implementation("org.eclipse.jgit:org.eclipse.jgit.ssh.apache:6.7.0.202309050840-r") | ||||||
|   implementation("com.vdurmont:semver4j:3.1.0") |   implementation("com.vdurmont:semver4j:3.1.0") | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -93,16 +93,28 @@ 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("selectBranch", JavaExec::class) { | tasks.register("resetReleaseBranch", JavaExec::class) { | ||||||
|   group = "release" |   group = "release" | ||||||
|   mainClass.set("scripts.release.SelectBranchKt") |   mainClass.set("scripts.release.ResetReleaseBranchKt") | ||||||
|   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("eapReleaseActions", JavaExec::class) { | tasks.register("pushChanges", 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.releaseEap.EapReleaseActionsKt") |   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 ?: "") | ||||||
| } | } | ||||||
|   | |||||||
| @@ -32,7 +32,6 @@ 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") ?: error("Missing YOUTRACK_TOKEN") |           val accessToken = System.getenv("YOUTRACK_TOKEN")!! | ||||||
|           BearerTokens(accessToken, "") |           BearerTokens(accessToken, "") | ||||||
|         } |         } | ||||||
|       } |       } | ||||||
|   | |||||||
							
								
								
									
										37
									
								
								scripts/src/main/kotlin/scripts/pushChanges.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										37
									
								
								scripts/src/main/kotlin/scripts/pushChanges.kt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,37 @@ | |||||||
|  | /* | ||||||
|  |  * 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") | ||||||
|  |   } | ||||||
|  | } | ||||||
| @@ -0,0 +1,54 @@ | |||||||
|  | /* | ||||||
|  |  * 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, ReleaseType.STABLE_NO_PATCH) |   val (lastVersion, objectId) = getVersion(projectDir, onlyStable = true) | ||||||
|   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, ReleaseType.ANY) |   val (lastVersion, _) = getVersion(projectDir, onlyStable = false) | ||||||
|  |  | ||||||
|   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, ReleaseType.ONLY_STABLE) |   val (lastVersion, _) = getVersion(projectDir, onlyStable = true) | ||||||
|  |  | ||||||
|   val nextVersion = when (releaseType) { |   val nextVersion = when (releaseType) { | ||||||
|     "major" -> lastVersion.nextMajor() |     "major" -> lastVersion.nextMajor() | ||||||
|   | |||||||
| @@ -8,45 +8,28 @@ | |||||||
|  |  | ||||||
| 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 = if (releaseType == "patch") { |   val newChangelog = changelog.replace("## To Be Released", newItem) | ||||||
|     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,8 +13,12 @@ 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 -> | ||||||
|     if (git.diff().call().isNotEmpty()) { |  | ||||||
|     git |     git | ||||||
|       .commit() |       .commit() | ||||||
|       .setAll(true) |       .setAll(true) | ||||||
| @@ -25,8 +29,5 @@ fun main(args: Array<String>) { | |||||||
|  |  | ||||||
|     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") |  | ||||||
|     } |  | ||||||
|   } |   } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -0,0 +1,38 @@ | |||||||
|  | /* | ||||||
|  |  * 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,7 +15,8 @@ fun main(args: Array<String>) { | |||||||
|  |  | ||||||
|   withGit(rootDir) { git -> |   withGit(rootDir) { git -> | ||||||
|     val branchName = when (releaseType) { |     val branchName = when (releaseType) { | ||||||
|       "major", "minor", "patch" -> "release" |       "major", "minor" -> "master" | ||||||
|  |       "patch" -> "release" | ||||||
|       else -> error("Unsupported release type: $releaseType") |       else -> error("Unsupported release type: $releaseType") | ||||||
|     } |     } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -9,7 +9,6 @@ | |||||||
| package scripts.release | package scripts.release | ||||||
|  |  | ||||||
| import com.vdurmont.semver4j.Semver | import com.vdurmont.semver4j.Semver | ||||||
| import org.eclipse.jgit.api.CreateBranchCommand |  | ||||||
| import org.eclipse.jgit.api.Git | import org.eclipse.jgit.api.Git | ||||||
| import org.eclipse.jgit.lib.ObjectId | import org.eclipse.jgit.lib.ObjectId | ||||||
| import org.eclipse.jgit.lib.Repository | import org.eclipse.jgit.lib.Repository | ||||||
| @@ -58,13 +57,7 @@ internal fun checkBranch(rootDir: String, releaseType: String) { | |||||||
|   } |   } | ||||||
| } | } | ||||||
|  |  | ||||||
| enum class ReleaseType { | internal fun getVersion(projectDir: String, onlyStable: Boolean): Pair<Semver, ObjectId> { | ||||||
|   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()) | ||||||
| @@ -81,24 +74,19 @@ internal fun getVersion(projectDir: String, releaseType: ReleaseType): Pair<Semv | |||||||
|   } |   } | ||||||
|     .sortedBy { it.first } |     .sortedBy { it.first } | ||||||
|  |  | ||||||
|   val version = when (releaseType) { |   val version = if (onlyStable) { | ||||||
|     ReleaseType.ANY -> versions.last() |     versions.last { it.first.isStable } | ||||||
|     ReleaseType.ONLY_STABLE -> versions.last { it.first.isStable } |   } else { | ||||||
|     ReleaseType.STABLE_NO_PATCH -> versions.last { it.first.isStable && it.first.patch == 0 } |     versions.last() | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   return version |   return version | ||||||
| } | } | ||||||
|  |  | ||||||
| internal fun Git.checkoutBranch(name: String) { | internal fun Git.checkoutBranch(name: String) { | ||||||
|   println("Checking out $name") |  | ||||||
|   val shouldCreateBranch = this.branchList().call().any { it.name == "refs/heads/$name" }.not() |   val shouldCreateBranch = this.branchList().call().any { it.name == "refs/heads/$name" }.not() | ||||||
|   val checkoutCommand = checkout() |   checkout() | ||||||
|     .setCreateBranch(shouldCreateBranch) |     .setCreateBranch(shouldCreateBranch) | ||||||
|     .setName(name) |     .setName(name) | ||||||
|   if (shouldCreateBranch) { |     .call() | ||||||
|     // Without starting point the branch will be created on HEAD. |  | ||||||
|     checkoutCommand.setStartPoint("origin/$name").setUpstreamMode(CreateBranchCommand.SetupUpstreamMode.TRACK) |  | ||||||
|   } |  | ||||||
|   checkoutCommand.call() |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,36 +0,0 @@ | |||||||
| /* |  | ||||||
|  * 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,8 +11,6 @@ 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 | ||||||
| @@ -23,10 +21,6 @@ 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() | ||||||
|  |  | ||||||
| @@ -65,59 +59,3 @@ 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,9 +12,4 @@ rootProject.name = 'IdeaVIM' | |||||||
| include 'vim-engine' | include 'vim-engine' | ||||||
| include 'scripts' | include 'scripts' | ||||||
| include 'annotation-processors' | include 'annotation-processors' | ||||||
| include 'tests:java-tests' |  | ||||||
| include 'tests:property-tests' |  | ||||||
| include 'tests:long-running-tests' |  | ||||||
| include 'tests:ui-ij-tests' |  | ||||||
| include 'tests:ui-py-tests' |  | ||||||
| include 'tests:ui-fixtures' |  | ||||||
|   | |||||||
| @@ -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.ProjectActivity | import com.intellij.openapi.startup.StartupActivity | ||||||
| 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,11 +20,16 @@ import com.maddyhome.idea.vim.newapi.globalIjOptions | |||||||
| /** | /** | ||||||
|  * @author Alex Plate |  * @author Alex Plate | ||||||
|  */ |  */ | ||||||
| internal class PluginStartup : ProjectActivity/*, LightEditCompatible*/ { | // This service should be migrated to ProjectActivity. But we should cariful because simple replacement | ||||||
|  | // 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 suspend fun execute(project: Project) { |   override fun runActivity(project: Project) { | ||||||
|     if (firstInitializationOccurred) return |     if (firstInitializationOccurred) return | ||||||
|     firstInitializationOccurred = true |     firstInitializationOccurred = true | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										79
									
								
								src/main/java/com/maddyhome/idea/vim/RegisterActions.java
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										79
									
								
								src/main/java/com/maddyhome/idea/vim/RegisterActions.java
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,79 @@ | |||||||
|  | /* | ||||||
|  |  * 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; | ||||||
|  |  | ||||||
|  | import com.intellij.openapi.application.ApplicationManager; | ||||||
|  | import com.intellij.openapi.extensions.ExtensionPointName; | ||||||
|  | import com.maddyhome.idea.vim.group.KeyGroup; | ||||||
|  | import com.maddyhome.idea.vim.handler.ActionBeanClass; | ||||||
|  | import com.maddyhome.idea.vim.handler.EditorActionHandlerBase; | ||||||
|  | import com.maddyhome.idea.vim.key.MappingOwner; | ||||||
|  | import com.maddyhome.idea.vim.newapi.IjVimActionsInitiator; | ||||||
|  | import org.jetbrains.annotations.NotNull; | ||||||
|  | import org.jetbrains.annotations.Nullable; | ||||||
|  |  | ||||||
|  | import javax.swing.*; | ||||||
|  | import java.awt.event.KeyEvent; | ||||||
|  |  | ||||||
|  | public class RegisterActions { | ||||||
|  |  | ||||||
|  |   public static final ExtensionPointName<ActionBeanClass> VIM_ACTIONS_EP = | ||||||
|  |     ExtensionPointName.create("IdeaVIM.vimAction"); | ||||||
|  |  | ||||||
|  |   /** | ||||||
|  |    * Register all the key/action mappings for the plugin. | ||||||
|  |    */ | ||||||
|  |   public static void registerActions() { | ||||||
|  |     registerVimCommandActions(); | ||||||
|  |     registerEmptyShortcuts(); | ||||||
|  |     registerEpListener(); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   private static void 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 static @Nullable EditorActionHandlerBase findAction(@NotNull String id) { | ||||||
|  |     return VIM_ACTIONS_EP.getExtensionList(ApplicationManager.getApplication()).stream() | ||||||
|  |         .filter(vimActionBean -> vimActionBean.getActionId().equals(id)).findFirst().map(ActionBeanClass::getInstance) | ||||||
|  |         .orElse(null); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   public static @NotNull EditorActionHandlerBase findActionOrDie(@NotNull String id) { | ||||||
|  |     EditorActionHandlerBase action = findAction(id); | ||||||
|  |     if (action == null) throw new RuntimeException("Action " + id + " is not registered"); | ||||||
|  |     return action; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   public static void unregisterActions() { | ||||||
|  |     KeyGroup keyGroup = VimPlugin.getKeyIfCreated(); | ||||||
|  |     if (keyGroup != null) { | ||||||
|  |       keyGroup.unregisterCommandActions(); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   private static void registerVimCommandActions() { | ||||||
|  |     KeyGroup parser = VimPlugin.getKey(); | ||||||
|  |     VIM_ACTIONS_EP.getExtensionList(ApplicationManager.getApplication()).stream().map(IjVimActionsInitiator::new) | ||||||
|  |         .forEach(parser::registerCommandAction); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   private static void registerEmptyShortcuts() { | ||||||
|  |     final KeyGroup parser = VimPlugin.getKey(); | ||||||
|  |  | ||||||
|  |     // The {char1} <BS> {char2} shortcut is handled directly by KeyHandler#handleKey, so doesn't have an action. But we | ||||||
|  |     // still need to register the shortcut, to make sure the editor doesn't swallow it. | ||||||
|  |     parser | ||||||
|  |       .registerShortcutWithoutAction(KeyStroke.getKeyStroke(KeyEvent.VK_BACK_SPACE, 0), MappingOwner.IdeaVim.System.INSTANCE); | ||||||
|  |   } | ||||||
|  | } | ||||||
| @@ -1,57 +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 |  | ||||||
|  |  | ||||||
| import com.maddyhome.idea.vim.action.EngineCommandProvider |  | ||||||
| import com.maddyhome.idea.vim.action.IntellijCommandProvider |  | ||||||
| import com.maddyhome.idea.vim.handler.EditorActionHandlerBase |  | ||||||
| import com.maddyhome.idea.vim.key.MappingOwner |  | ||||||
| import java.awt.event.KeyEvent |  | ||||||
| import javax.swing.KeyStroke |  | ||||||
|  |  | ||||||
| public object RegisterActions { |  | ||||||
|   /** |  | ||||||
|    * Register all the key/action mappings for the plugin. |  | ||||||
|    */ |  | ||||||
|   @JvmStatic |  | ||||||
|   public fun registerActions() { |  | ||||||
|     registerVimCommandActions() |  | ||||||
|     registerEmptyShortcuts() // todo most likely it is not needed |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   public fun findAction(id: String): EditorActionHandlerBase? { |  | ||||||
|     val commandBean = EngineCommandProvider.getCommands().firstOrNull { it.actionId == id } |  | ||||||
|       ?: IntellijCommandProvider.getCommands().firstOrNull { it.actionId == id } ?: return null |  | ||||||
|     return commandBean.instance |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   public fun findActionOrDie(id: String): EditorActionHandlerBase { |  | ||||||
|     return findAction(id) ?: throw RuntimeException("Action $id is not registered") |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   @JvmStatic |  | ||||||
|   public fun unregisterActions() { |  | ||||||
|     val keyGroup = VimPlugin.getKeyIfCreated() |  | ||||||
|     keyGroup?.unregisterCommandActions() |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   private fun registerVimCommandActions() { |  | ||||||
|     val parser = VimPlugin.getKey() |  | ||||||
|     EngineCommandProvider.getCommands().forEach { parser.registerCommandAction(it) } |  | ||||||
|     IntellijCommandProvider.getCommands().forEach { parser.registerCommandAction(it) } |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   private fun registerEmptyShortcuts() { |  | ||||||
|     val parser = VimPlugin.getKey() |  | ||||||
|  |  | ||||||
|     // The {char1} <BS> {char2} shortcut is handled directly by KeyHandler#handleKey, so doesn't have an action. But we |  | ||||||
|     // still need to register the shortcut, to make sure the editor doesn't swallow it. |  | ||||||
|     parser |  | ||||||
|       .registerShortcutWithoutAction(KeyStroke.getKeyStroke(KeyEvent.VK_BACK_SPACE, 0), MappingOwner.IdeaVim.System) |  | ||||||
|   } |  | ||||||
| } |  | ||||||
| @@ -39,9 +39,11 @@ 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; | ||||||
| @@ -115,6 +117,12 @@ 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()); | ||||||
|   } |   } | ||||||
| @@ -187,6 +195,13 @@ 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); | ||||||
|   } |   } | ||||||
| @@ -204,10 +219,6 @@ 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; | ||||||
|  |  | ||||||
| @@ -221,13 +232,7 @@ public class VimPlugin implements PersistentStateComponent<Element>, Disposable | |||||||
|       getInstance().turnOnPlugin(); |       getInstance().turnOnPlugin(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     if (enabled) { |     StatusBarIconFactory.Companion.updateIcon(); | ||||||
|       VimInjectorKt.getInjector().getListenersNotifier().notifyPluginTurnedOn(); |  | ||||||
|     } else { |  | ||||||
|       VimInjectorKt.getInjector().getListenersNotifier().notifyPluginTurnedOff(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     StatusBarIconFactory.Util.INSTANCE.updateIcon(); |  | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   public static String getMessage() { |   public static String getMessage() { | ||||||
| @@ -259,8 +264,7 @@ 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()) { | ||||||
|         turnOnPlugin(); |         application.invokeAndWait(this::turnOnPlugin); | ||||||
|         //application.invokeAndWait(this::turnOnPlugin); |  | ||||||
|       } |       } | ||||||
|       else { |       else { | ||||||
|         application.invokeLater(this::turnOnPlugin); |         application.invokeLater(this::turnOnPlugin); | ||||||
| @@ -302,6 +306,11 @@ public class VimPlugin implements PersistentStateComponent<Element>, Disposable | |||||||
|    * This is required to ensure that all options are correctly initialised and registered. Must be before any commands |    * This is required to ensure that all options are correctly initialised and registered. Must be before any commands | ||||||
|    * are executed.</li> |    * are executed.</li> | ||||||
|    * <li>~/.ideavimrc execution<br> |    * <li>~/.ideavimrc execution<br> | ||||||
|  |    * <ul> | ||||||
|  |    * <li>4.1 executes commands from the .ideavimrc file and 4.2 initializes extensions.</li> | ||||||
|  |    * <li>4.1 MUST BE BEFORE 4.2. This is a flow of vim/IdeaVim initialization, firstly .ideavimrc is executed and then | ||||||
|  |    * the extensions are initialized.</li> | ||||||
|  |    * </ul> | ||||||
|    * </li> |    * </li> | ||||||
|    * <li>Components initialization<br> |    * <li>Components initialization<br> | ||||||
|    * This should happen after ideavimrc execution because VimListenerManager accesses `number` option |    * This should happen after ideavimrc execution because VimListenerManager accesses `number` option | ||||||
| @@ -330,9 +339,13 @@ public class VimPlugin implements PersistentStateComponent<Element>, Disposable | |||||||
|     VimInjectorKt.getInjector().getOptionGroup().initialiseOptions(); |     VimInjectorKt.getInjector().getOptionGroup().initialiseOptions(); | ||||||
|  |  | ||||||
|     // 4) ~/.ideavimrc execution |     // 4) ~/.ideavimrc execution | ||||||
|  |     // 4.1) Execute ~/.ideavimrc | ||||||
|     // Evaluate in the context of the fallback window, to capture local option state, to copy to the first editor window |     // Evaluate in the context of the fallback window, to capture local option state, to copy to the first editor window | ||||||
|     registerIdeavimrc(VimInjectorKt.getInjector().getFallbackWindow()); |     registerIdeavimrc(VimInjectorKt.getInjector().getFallbackWindow()); | ||||||
|  |  | ||||||
|  |     // 4.2) Initialize extensions. Always after 4.1 | ||||||
|  |     VimExtensionRegistrar.enableDelayedExtensions(); | ||||||
|  |  | ||||||
|     // Turing on should be performed after all commands registration |     // Turing on should be performed after all commands registration | ||||||
|     getSearch().turnOn(); |     getSearch().turnOn(); | ||||||
|     VimListenerManager.INSTANCE.turnOn(); |     VimListenerManager.INSTANCE.turnOn(); | ||||||
|   | |||||||
| @@ -28,11 +28,8 @@ 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 |  | ||||||
|  */ |  */ | ||||||
| public class VimTypedActionHandler(origHandler: TypedActionHandler) : TypedActionHandlerEx { | internal 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 | ||||||
|  |  | ||||||
| @@ -89,7 +86,7 @@ public class VimTypedActionHandler(origHandler: TypedActionHandler) : TypedActio | |||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   internal companion object { |   companion object { | ||||||
|     private val LOG = logger<VimTypedActionHandler>() |     private val LOG = logger<VimTypedActionHandler>() | ||||||
|   } |   } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -7,8 +7,6 @@ | |||||||
|  */ |  */ | ||||||
| package com.maddyhome.idea.vim.action | package com.maddyhome.idea.vim.action | ||||||
|  |  | ||||||
| import com.intellij.vim.annotations.CommandOrMotion |  | ||||||
| import com.intellij.vim.annotations.Mode |  | ||||||
| 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 | ||||||
| import com.maddyhome.idea.vim.api.VimEditor | import com.maddyhome.idea.vim.api.VimEditor | ||||||
| @@ -16,7 +14,6 @@ import com.maddyhome.idea.vim.command.Command | |||||||
| 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 | ||||||
|  |  | ||||||
| @CommandOrMotion(keys = [":"], modes = [Mode.NORMAL, Mode.VISUAL, Mode.OP_PENDING]) |  | ||||||
| internal class ExEntryAction : VimActionHandler.SingleExecution() { | internal class ExEntryAction : VimActionHandler.SingleExecution() { | ||||||
|   override val type: Command.Type = Command.Type.OTHER_READONLY |   override val type: Command.Type = Command.Type.OTHER_READONLY | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,13 +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 |  | ||||||
|  |  | ||||||
| public object IntellijCommandProvider : CommandProvider { |  | ||||||
|   override val commandListFileName: String = "intellij_commands.json" |  | ||||||
| } |  | ||||||
| @@ -14,10 +14,14 @@ 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.AnActionWrapper | import com.intellij.openapi.actionSystem.AnActionWrapper | ||||||
|  | import com.intellij.openapi.actionSystem.IdeActions | ||||||
|  | import com.intellij.openapi.actionSystem.KeyboardShortcut | ||||||
| import com.intellij.openapi.actionSystem.PlatformDataKeys | import com.intellij.openapi.actionSystem.PlatformDataKeys | ||||||
| import com.intellij.openapi.application.invokeLater | import com.intellij.openapi.application.invokeLater | ||||||
| import com.intellij.openapi.diagnostic.logger | import com.intellij.openapi.diagnostic.logger | ||||||
| import com.intellij.openapi.editor.Editor | import com.intellij.openapi.editor.Editor | ||||||
|  | import com.intellij.openapi.editor.actionSystem.EditorActionManager | ||||||
|  | import com.intellij.openapi.keymap.KeymapManager | ||||||
| import com.intellij.openapi.progress.ProcessCanceledException | import com.intellij.openapi.progress.ProcessCanceledException | ||||||
| import com.intellij.openapi.project.DumbAware | import com.intellij.openapi.project.DumbAware | ||||||
| import com.intellij.openapi.util.Key | import com.intellij.openapi.util.Key | ||||||
| @@ -55,17 +59,9 @@ 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 |  | ||||||
|  */ |  */ | ||||||
| public class VimShortcutKeyAction : AnAction(), DumbAware/*, LightEditCompatible*/ { | internal class VimShortcutKeyAction : AnAction(), DumbAware/*, LightEditCompatible*/ { | ||||||
|   private val traceTime: Boolean |   private val traceTime = injector.globalOptions().ideatracetime | ||||||
|     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") | ||||||
| @@ -99,30 +95,28 @@ public class VimShortcutKeyAction : AnAction(), DumbAware/*, LightEditCompatible | |||||||
|  |  | ||||||
|   // 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 = ActionUpdateThread.EDT |   override fun getActionUpdateThread() = 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 | ||||||
|     val keyStroke = getKeyStroke(e) |     val actionEnableStatus = isEnabled(e) | ||||||
|     val actionEnableStatus = isEnabled(e, keyStroke) |  | ||||||
|     e.presentation.isEnabled = actionEnableStatus.isEnabled |     e.presentation.isEnabled = actionEnableStatus.isEnabled | ||||||
|     actionEnableStatus.printLog(keyStroke) |     actionEnableStatus.printLog() | ||||||
|     if (start != null) { |     if (start != null) { | ||||||
|  |       val keyStroke = getKeyStroke(e) | ||||||
|       val duration = System.currentTimeMillis() - start |       val duration = System.currentTimeMillis() - start | ||||||
|       LOG.info("VimShortcut update '$keyStroke': $duration ms") |       LOG.info("VimShortcut update '$keyStroke': $duration ms") | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   private fun isEnabled(e: AnActionEvent, keyStroke: KeyStroke?): ActionEnableStatus { |   private fun isEnabled(e: AnActionEvent): ActionEnableStatus { | ||||||
|     if (VimPlugin.isNotEnabled()) return ActionEnableStatus.no("IdeaVim is disabled", LogLevel.DEBUG) |     if (!VimPlugin.isEnabled()) return ActionEnableStatus.no("IdeaVim is disabled", LogLevel.DEBUG) | ||||||
|     val editor = getEditor(e) |     val editor = getEditor(e) | ||||||
|  |     val keyStroke = getKeyStroke(e) | ||||||
|     if (editor != null && keyStroke != null) { |     if (editor != null && keyStroke != null) { | ||||||
|       if (enableOctopus) { |       if (enableOctopus) { | ||||||
|         if (isOctopusEnabled(keyStroke, editor)) { |         if (isOctopusEnabled(keyStroke, editor)) { | ||||||
|           return ActionEnableStatus.no( |           return ActionEnableStatus.no("Octopus handler is enabled", LogLevel.DEBUG) | ||||||
|             "Processing VimShortcutKeyAction for the key that is used in the octopus handler", |  | ||||||
|             LogLevel.ERROR |  | ||||||
|           ) |  | ||||||
|         } |         } | ||||||
|       } |       } | ||||||
|       if (editor.isIdeaVimDisabledHere) { |       if (editor.isIdeaVimDisabledHere) { | ||||||
| @@ -170,6 +164,18 @@ public class VimShortcutKeyAction : AnAction(), DumbAware/*, LightEditCompatible | |||||||
|         return ActionEnableStatus.no("App code template is active", LogLevel.INFO) |         return ActionEnableStatus.no("App code template is active", LogLevel.INFO) | ||||||
|       } |       } | ||||||
|        |        | ||||||
|  |       if (keyCode == KeyEvent.VK_LEFT || keyCode == KeyEvent.VK_RIGHT || keyCode == KeyEvent.VK_UP || keyCode == KeyEvent.VK_DOWN || keyCode == KeyEvent.VK_HOME || keyCode == KeyEvent.VK_END) { | ||||||
|  |         return ActionEnableStatus.no("Special keys", LogLevel.INFO) | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       val nextTemplateVariableShortcuts = KeymapManager.getInstance().activeKeymap.getShortcuts(IdeActions.ACTION_EDITOR_NEXT_TEMPLATE_VARIABLE) | ||||||
|  |       if (nextTemplateVariableShortcuts.any { it is KeyboardShortcut && it.firstKeyStroke == keyStroke }) { | ||||||
|  |         val handler = EditorActionManager.getInstance().getActionHandler(IdeActions.ACTION_EDITOR_NEXT_TEMPLATE_VARIABLE) | ||||||
|  |         if (handler.isEnabled(editor, null, e.dataContext)) { | ||||||
|  |           return ActionEnableStatus.no("Next template variable or finish in-place refactoring", LogLevel.INFO) | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |        | ||||||
|       if (editor.inInsertMode) { |       if (editor.inInsertMode) { | ||||||
|         if (keyCode == KeyEvent.VK_TAB) { |         if (keyCode == KeyEvent.VK_TAB) { | ||||||
|           // TODO: This stops VimEditorTab seeing <Tab> in insert mode and correctly scrolling the view |           // TODO: This stops VimEditorTab seeing <Tab> in insert mode and correctly scrolling the view | ||||||
| @@ -235,9 +241,9 @@ public class VimShortcutKeyAction : AnAction(), DumbAware/*, LightEditCompatible | |||||||
|   /** |   /** | ||||||
|    * 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 and getDefaultKeyStroke returns null |    * the event is already consumed | ||||||
|    */ |    */ | ||||||
|   private var keyStrokeCache: Pair<Long?, KeyStroke?> = null to null |   private var keyStrokeCache: Pair<KeyEvent?, KeyStroke?> = null to null | ||||||
|  |  | ||||||
|   private fun getKeyStroke(e: AnActionEvent): KeyStroke? { |   private fun getKeyStroke(e: AnActionEvent): KeyStroke? { | ||||||
|     val inputEvent = e.inputEvent |     val inputEvent = e.inputEvent | ||||||
| @@ -245,9 +251,9 @@ public class VimShortcutKeyAction : AnAction(), DumbAware/*, LightEditCompatible | |||||||
|       val defaultKeyStroke = KeyStrokeAdapter.getDefaultKeyStroke(inputEvent) |       val defaultKeyStroke = KeyStrokeAdapter.getDefaultKeyStroke(inputEvent) | ||||||
|       val strokeCache = keyStrokeCache |       val strokeCache = keyStrokeCache | ||||||
|       if (defaultKeyStroke != null) { |       if (defaultKeyStroke != null) { | ||||||
|         keyStrokeCache = inputEvent.`when` to defaultKeyStroke |         keyStrokeCache = inputEvent to defaultKeyStroke | ||||||
|         return defaultKeyStroke |         return defaultKeyStroke | ||||||
|       } else if (strokeCache.first == inputEvent.`when`) { |       } else if (strokeCache.first === inputEvent) { | ||||||
|         keyStrokeCache = null to null |         keyStrokeCache = null to null | ||||||
|         return strokeCache.second |         return strokeCache.second | ||||||
|       } |       } | ||||||
| @@ -280,7 +286,7 @@ public class VimShortcutKeyAction : AnAction(), DumbAware/*, LightEditCompatible | |||||||
|       .toSet() |       .toSet() | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   internal companion object { |   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)) | ||||||
| @@ -371,12 +377,10 @@ private class ActionEnableStatus( | |||||||
|   val message: String, |   val message: String, | ||||||
|   val logLevel: LogLevel, |   val logLevel: LogLevel, | ||||||
| ) { | ) { | ||||||
|   fun printLog(keyStroke: KeyStroke?) { |   fun printLog() { | ||||||
|     val message = "IdeaVim keys are enabled = $isEnabled for key '$keyStroke': $message" |  | ||||||
|     when (logLevel) { |     when (logLevel) { | ||||||
|       LogLevel.INFO -> LOG.info(message) |       LogLevel.INFO -> LOG.info("IdeaVim keys are enabled = $isEnabled: $message") | ||||||
|       LogLevel.DEBUG -> LOG.debug(message) |       LogLevel.DEBUG -> LOG.debug("IdeaVim keys are enabled = $isEnabled: $message") | ||||||
|       LogLevel.ERROR -> LOG.error(message) |  | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
| @@ -389,5 +393,5 @@ private class ActionEnableStatus( | |||||||
| } | } | ||||||
|  |  | ||||||
| private enum class LogLevel { | private enum class LogLevel { | ||||||
|   DEBUG, INFO, ERROR, |   DEBUG, INFO, | ||||||
| } | } | ||||||
| @@ -7,8 +7,6 @@ | |||||||
|  */ |  */ | ||||||
| package com.maddyhome.idea.vim.action.change | package com.maddyhome.idea.vim.action.change | ||||||
|  |  | ||||||
| import com.intellij.vim.annotations.CommandOrMotion |  | ||||||
| import com.intellij.vim.annotations.Mode |  | ||||||
| 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.ExecutionContext | import com.maddyhome.idea.vim.api.ExecutionContext | ||||||
| @@ -18,7 +16,9 @@ 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 | ||||||
| @@ -26,9 +26,10 @@ 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 com.maddyhome.idea.vim.state.mode.SelectionType | import java.util.* | ||||||
|  |  | ||||||
| // 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 { | ||||||
| @@ -46,7 +47,6 @@ private fun doOperatorAction(editor: VimEditor, context: ExecutionContext, textR | |||||||
|   return result |   return result | ||||||
| } | } | ||||||
|  |  | ||||||
| @CommandOrMotion(keys = ["g@"], modes = [Mode.NORMAL]) |  | ||||||
| internal class OperatorAction : VimActionHandler.SingleExecution() { | internal class OperatorAction : VimActionHandler.SingleExecution() { | ||||||
|   override val type: Command.Type = Command.Type.OTHER_SELF_SYNCHRONIZED |   override val type: Command.Type = Command.Type.OTHER_SELF_SYNCHRONIZED | ||||||
|  |  | ||||||
| @@ -97,10 +97,11 @@ internal class OperatorAction : VimActionHandler.SingleExecution() { | |||||||
|   } |   } | ||||||
| } | } | ||||||
|  |  | ||||||
| @CommandOrMotion(keys = ["g@"], modes = [Mode.VISUAL]) |  | ||||||
| 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, | ||||||
|   | |||||||
| @@ -7,8 +7,6 @@ | |||||||
|  */ |  */ | ||||||
| package com.maddyhome.idea.vim.action.change | package com.maddyhome.idea.vim.action.change | ||||||
|  |  | ||||||
| import com.intellij.vim.annotations.CommandOrMotion |  | ||||||
| import com.intellij.vim.annotations.Mode |  | ||||||
| import com.intellij.openapi.command.CommandProcessor | import com.intellij.openapi.command.CommandProcessor | ||||||
| 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 | ||||||
| @@ -20,7 +18,6 @@ import com.maddyhome.idea.vim.handler.VimActionHandler | |||||||
| 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 | ||||||
|  |  | ||||||
| @CommandOrMotion(keys = ["."], modes = [Mode.NORMAL]) |  | ||||||
| internal class RepeatChangeAction : VimActionHandler.SingleExecution() { | internal class RepeatChangeAction : VimActionHandler.SingleExecution() { | ||||||
|   override val type: Command.Type = Command.Type.OTHER_WRITABLE |   override val type: Command.Type = Command.Type.OTHER_WRITABLE | ||||||
|  |  | ||||||
|   | |||||||
| @@ -7,8 +7,6 @@ | |||||||
|  */ |  */ | ||||||
| package com.maddyhome.idea.vim.action.change.delete | package com.maddyhome.idea.vim.action.change.delete | ||||||
|  |  | ||||||
| import com.intellij.vim.annotations.CommandOrMotion |  | ||||||
| import com.intellij.vim.annotations.Mode |  | ||||||
| import com.maddyhome.idea.vim.api.ExecutionContext | import com.maddyhome.idea.vim.api.ExecutionContext | ||||||
| import com.maddyhome.idea.vim.api.VimCaret | import com.maddyhome.idea.vim.api.VimCaret | ||||||
| import com.maddyhome.idea.vim.api.VimEditor | import com.maddyhome.idea.vim.api.VimEditor | ||||||
| @@ -19,7 +17,6 @@ import com.maddyhome.idea.vim.command.OperatorArguments | |||||||
| import com.maddyhome.idea.vim.handler.ChangeEditorActionHandler | import com.maddyhome.idea.vim.handler.ChangeEditorActionHandler | ||||||
| import com.maddyhome.idea.vim.newapi.ijOptions | import com.maddyhome.idea.vim.newapi.ijOptions | ||||||
|  |  | ||||||
| @CommandOrMotion(keys = ["gJ"], modes = [Mode.NORMAL]) |  | ||||||
| public class DeleteJoinLinesAction : ChangeEditorActionHandler.ConditionalSingleExecution() { | public class DeleteJoinLinesAction : ChangeEditorActionHandler.ConditionalSingleExecution() { | ||||||
|   override val type: Command.Type = Command.Type.DELETE |   override val type: Command.Type = Command.Type.DELETE | ||||||
|   override fun runAsMulticaret( |   override fun runAsMulticaret( | ||||||
|   | |||||||
| @@ -7,8 +7,6 @@ | |||||||
|  */ |  */ | ||||||
| package com.maddyhome.idea.vim.action.change.delete | package com.maddyhome.idea.vim.action.change.delete | ||||||
|  |  | ||||||
| import com.intellij.vim.annotations.CommandOrMotion |  | ||||||
| import com.intellij.vim.annotations.Mode |  | ||||||
| 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 | ||||||
| @@ -18,7 +16,6 @@ import com.maddyhome.idea.vim.command.OperatorArguments | |||||||
| import com.maddyhome.idea.vim.handler.ChangeEditorActionHandler | import com.maddyhome.idea.vim.handler.ChangeEditorActionHandler | ||||||
| import com.maddyhome.idea.vim.newapi.ijOptions | import com.maddyhome.idea.vim.newapi.ijOptions | ||||||
|  |  | ||||||
| @CommandOrMotion(keys = ["J"], modes = [Mode.NORMAL]) |  | ||||||
| public class DeleteJoinLinesSpacesAction : ChangeEditorActionHandler.SingleExecution() { | public class DeleteJoinLinesSpacesAction : ChangeEditorActionHandler.SingleExecution() { | ||||||
|   override val type: Command.Type = Command.Type.DELETE |   override val type: Command.Type = Command.Type.DELETE | ||||||
|  |  | ||||||
|   | |||||||
| @@ -7,25 +7,27 @@ | |||||||
|  */ |  */ | ||||||
| package com.maddyhome.idea.vim.action.change.delete | package com.maddyhome.idea.vim.action.change.delete | ||||||
|  |  | ||||||
| import com.intellij.vim.annotations.CommandOrMotion |  | ||||||
| import com.intellij.vim.annotations.Mode |  | ||||||
| import com.maddyhome.idea.vim.api.ExecutionContext | import com.maddyhome.idea.vim.api.ExecutionContext | ||||||
| import com.maddyhome.idea.vim.api.VimCaret | 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 | ||||||
|  */ |  */ | ||||||
| @CommandOrMotion(keys = ["gJ"], modes = [Mode.VISUAL]) |  | ||||||
| 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, | ||||||
|   | |||||||
| @@ -7,25 +7,27 @@ | |||||||
|  */ |  */ | ||||||
| package com.maddyhome.idea.vim.action.change.delete | package com.maddyhome.idea.vim.action.change.delete | ||||||
|  |  | ||||||
| import com.intellij.vim.annotations.CommandOrMotion |  | ||||||
| import com.intellij.vim.annotations.Mode |  | ||||||
| import com.maddyhome.idea.vim.api.ExecutionContext | import com.maddyhome.idea.vim.api.ExecutionContext | ||||||
| import com.maddyhome.idea.vim.api.VimCaret | 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 | ||||||
|  */ |  */ | ||||||
| @CommandOrMotion(keys = ["J"], modes = [Mode.VISUAL]) |  | ||||||
| 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, | ||||||
|   | |||||||
| @@ -9,8 +9,7 @@ | |||||||
| package com.maddyhome.idea.vim.action.editor | package com.maddyhome.idea.vim.action.editor | ||||||
|  |  | ||||||
| import com.intellij.openapi.actionSystem.IdeActions | import com.intellij.openapi.actionSystem.IdeActions | ||||||
| import com.intellij.vim.annotations.CommandOrMotion | import com.maddyhome.idea.vim.action.ComplicatedKeysAction | ||||||
| import com.intellij.vim.annotations.Mode |  | ||||||
| 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 | ||||||
| @@ -20,37 +19,53 @@ 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]) | 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]) | 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]) | 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]) | 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]) | 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) | ||||||
| } | } | ||||||
|  |  | ||||||
| @CommandOrMotion(keys = ["K"], modes = [Mode.NORMAL]) |  | ||||||
| internal class VimQuickJavaDoc : VimActionHandler.SingleExecution() { | internal class VimQuickJavaDoc : VimActionHandler.SingleExecution() { | ||||||
|   override val type: Command.Type = Command.Type.OTHER_READONLY |   override val type: Command.Type = Command.Type.OTHER_READONLY | ||||||
|  |  | ||||||
|   | |||||||
| @@ -7,9 +7,8 @@ | |||||||
|  */ |  */ | ||||||
| package com.maddyhome.idea.vim.action.ex | package com.maddyhome.idea.vim.action.ex | ||||||
|  |  | ||||||
| import com.intellij.vim.annotations.CommandOrMotion |  | ||||||
| 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 | ||||||
| @@ -17,14 +16,17 @@ 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 | ||||||
|  * |  * | ||||||
|  * 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]) | 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) { | ||||||
|         is com.maddyhome.idea.vim.state.mode.Mode.CMD_LINE -> CommandState.Mode.CMD_LINE |         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,7 +9,6 @@ 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 | ||||||
| @@ -18,6 +17,7 @@ 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,7 +26,6 @@ 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 | ||||||
| @@ -39,9 +38,6 @@ 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( | ||||||
| @@ -144,12 +140,10 @@ 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) { | ||||||
| @@ -157,13 +151,11 @@ 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,11 +53,6 @@ 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) | ||||||
| @@ -87,31 +82,6 @@ internal object VimExtensionRegistrar : VimExtensionRegistrator { | |||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   /** |  | ||||||
|    * During vim initialization process, it firstly loads the .vimrc file, then executes scripts from the plugins folder. |  | ||||||
|    * This practically means that the .vimrc file is initialized first, then the plugins are loaded. |  | ||||||
|    * See `:h initialization` |  | ||||||
|    * |  | ||||||
|    * In IdeaVim we don't have a separate plugins folder to load it after .ideavimrc load. However, we can collect |  | ||||||
|    *   the list of plugins mentioned in the .ideavimrc and load them after .ideavimrc execution is finished. |  | ||||||
|    * |  | ||||||
|    * Why this matters? Because this affects the order of commands are executed. For example: |  | ||||||
|    * ``` |  | ||||||
|    * plug 'tommcdo/vim-exchange' |  | ||||||
|    * let g:exchange_no_mappings=1 |  | ||||||
|    * ``` |  | ||||||
|    * Here the user will expect that the exchange plugin won't have default mappings. However, if we load vim-exchange |  | ||||||
|    *    immediately, this variable won't be initialized at the moment of plugin initialization. |  | ||||||
|    * |  | ||||||
|    * There is also a tricky case for mappings override: |  | ||||||
|    * ``` |  | ||||||
|    * plug 'tommcdo/vim-exchange' |  | ||||||
|    * map X <Plug>(ExchangeLine) |  | ||||||
|    * ``` |  | ||||||
|    * For this case, a plugin with a good implementation detects that there is already a defined mapping for |  | ||||||
|    *   `<Plug>(ExchangeLine)` and doesn't register the default cxx mapping. However, such detection requires the mapping |  | ||||||
|    *   to be defined before the plugin initialization. |  | ||||||
|    */ |  | ||||||
|   @JvmStatic |   @JvmStatic | ||||||
|   fun enableDelayedExtensions() { |   fun enableDelayedExtensions() { | ||||||
|     delayedExtensionEnabling.forEach { |     delayedExtensionEnabling.forEach { | ||||||
|   | |||||||
| @@ -11,7 +11,6 @@ import com.intellij.codeInsight.actions.AsyncActionExecutionService | |||||||
| import com.intellij.openapi.actionSystem.IdeActions | import com.intellij.openapi.actionSystem.IdeActions | ||||||
| import com.intellij.openapi.application.runWriteAction | import com.intellij.openapi.application.runWriteAction | ||||||
| import com.intellij.openapi.editor.Editor | import com.intellij.openapi.editor.Editor | ||||||
| import com.intellij.openapi.project.Project |  | ||||||
| import com.intellij.openapi.util.Ref | import com.intellij.openapi.util.Ref | ||||||
| import com.intellij.psi.PsiComment | import com.intellij.psi.PsiComment | ||||||
| import com.intellij.psi.PsiElement | import com.intellij.psi.PsiElement | ||||||
| @@ -75,24 +74,17 @@ internal class CommentaryExtension : VimExtension { | |||||||
|           listOf(IdeActions.ACTION_COMMENT_BLOCK, IdeActions.ACTION_COMMENT_LINE) |           listOf(IdeActions.ACTION_COMMENT_BLOCK, IdeActions.ACTION_COMMENT_LINE) | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         val project = editor.ij.project!! |         val res = Ref.create<Boolean>(true) | ||||||
|         val callback = { afterCommenting(mode, editor, resetCaret, range) } |         AsyncActionExecutionService.getInstance(editor.ij.project!!).withExecutionAfterAction(actions[0], { | ||||||
|         actions.any { executeActionWithCallbackOnSuccess(it, project, context, callback) } |           res.set(injector.actionExecutor.executeAction(actions[0], context)) | ||||||
|  |         }, { afterCommenting(mode, editor, resetCaret, range) }) | ||||||
|  |         if (!res.get()) { | ||||||
|  |           AsyncActionExecutionService.getInstance(editor.ij.project!!).withExecutionAfterAction(actions[1], { | ||||||
|  |             res.set(injector.actionExecutor.executeAction(actions[1], context)) | ||||||
|  |           }, { afterCommenting(mode, editor, resetCaret, range) }) | ||||||
|         } |         } | ||||||
|  |         res.get() | ||||||
|       } |       } | ||||||
|  |  | ||||||
|     private fun executeActionWithCallbackOnSuccess( |  | ||||||
|       action: String, |  | ||||||
|       project: Project, |  | ||||||
|       context: ExecutionContext, |  | ||||||
|       callback: () -> Unit, |  | ||||||
|     ): Boolean { |  | ||||||
|       val res = Ref.create<Boolean>(false) |  | ||||||
|       AsyncActionExecutionService.getInstance(project).withExecutionAfterAction( |  | ||||||
|         action, |  | ||||||
|         { res.set(injector.actionExecutor.executeAction(action, context)) }, |  | ||||||
|         { if (res.get()) callback() }) |  | ||||||
|       return res.get() |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private fun afterCommenting( |     private fun afterCommenting( | ||||||
| @@ -156,6 +148,11 @@ 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) | ||||||
|   | |||||||
| @@ -217,8 +217,6 @@ private object FileTypePatterns { | |||||||
|  |  | ||||||
|     return if (fileTypeName in htmlLikeFileTypes) { |     return if (fileTypeName in htmlLikeFileTypes) { | ||||||
|       this.htmlPatterns |       this.htmlPatterns | ||||||
|     } else if (fileTypeName == "JAVA" || fileExtension == "java") { |  | ||||||
|       this.javaPatterns |  | ||||||
|     } else if (fileTypeName == "Ruby" || fileExtension == "rb") { |     } else if (fileTypeName == "Ruby" || fileExtension == "rb") { | ||||||
|       this.rubyPatterns |       this.rubyPatterns | ||||||
|     } else if (fileTypeName == "RHTML" || fileExtension == "erb") { |     } else if (fileTypeName == "RHTML" || fileExtension == "erb") { | ||||||
| @@ -244,7 +242,6 @@ private object FileTypePatterns { | |||||||
|   ) |   ) | ||||||
|  |  | ||||||
|   private val htmlPatterns = createHtmlPatterns() |   private val htmlPatterns = createHtmlPatterns() | ||||||
|   private val javaPatterns = createJavaPatterns() |  | ||||||
|   private val rubyPatterns = createRubyPatterns() |   private val rubyPatterns = createRubyPatterns() | ||||||
|   private val rubyAndHtmlPatterns = rubyPatterns + htmlPatterns |   private val rubyAndHtmlPatterns = rubyPatterns + htmlPatterns | ||||||
|   private val phpPatterns = createPhpPatterns() |   private val phpPatterns = createPhpPatterns() | ||||||
| @@ -274,14 +271,6 @@ private object FileTypePatterns { | |||||||
|       ) |       ) | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   private fun createJavaPatterns(): LanguagePatterns { |  | ||||||
|     return ( |  | ||||||
|         LanguagePatterns("\\b(?<!else\\s+)if\\b", "\\belse\\s+if\\b", "\\belse(?!\\s+if)\\b") + |  | ||||||
|           LanguagePatterns("\\bdo\\b", "\\bwhile\\b") + |  | ||||||
|           LanguagePatterns("\\btry\\b", "\\bcatch\\b", "\\bfinally\\b") |  | ||||||
|       ) |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   private fun createRubyPatterns(): LanguagePatterns { |   private fun createRubyPatterns(): LanguagePatterns { | ||||||
|     // Original patterns: https://github.com/vim/vim/blob/master/runtime/ftplugin/ruby.vim |     // Original patterns: https://github.com/vim/vim/blob/master/runtime/ftplugin/ruby.vim | ||||||
|     // We use non-capturing groups (?:) since we don't need any back refs. The \\b marker takes care of word boundaries. |     // We use non-capturing groups (?:) since we don't need any back refs. The \\b marker takes care of word boundaries. | ||||||
|   | |||||||
| @@ -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 = injector.searchHelper.findAll(IjVimEditor(editor), pattern, 0, -1, false) |       val matches = SearchHelper.findAll(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 injector.searchHelper.findPattern(IjVimEditor(editor), makePattern(text, whole), startOffset, 1, searchOptions)?.startOffset ?: -1 |     return SearchHelper.findPattern(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 { | ||||||
|   | |||||||
| @@ -502,13 +502,6 @@ internal class NerdTree : VimExtension { | |||||||
|         } |         } | ||||||
|       }, |       }, | ||||||
|     ) |     ) | ||||||
|      |  | ||||||
|     for (c in ('a'..'z') + ('A'..'Z')) { |  | ||||||
|       val ks = KeyStroke.getKeyStroke(c) |  | ||||||
|       if (ks !in actionsRoot) { |  | ||||||
|         registerCommand(c.toString(), NerdAction.Code { _, _, _ -> }) |  | ||||||
|       } |  | ||||||
|     } |  | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   companion object { |   companion object { | ||||||
|   | |||||||
| @@ -11,8 +11,8 @@ package com.maddyhome.idea.vim.extension.paragraphmotion | |||||||
| import com.intellij.openapi.editor.Caret | import com.intellij.openapi.editor.Caret | ||||||
| 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.getLineEndForOffset |  | ||||||
| import com.maddyhome.idea.vim.api.injector | import com.maddyhome.idea.vim.api.injector | ||||||
|  | import com.maddyhome.idea.vim.api.normalizeOffset | ||||||
| import com.maddyhome.idea.vim.command.MappingMode | import com.maddyhome.idea.vim.command.MappingMode | ||||||
| import com.maddyhome.idea.vim.command.OperatorArguments | import com.maddyhome.idea.vim.command.OperatorArguments | ||||||
| import com.maddyhome.idea.vim.extension.ExtensionHandler | import com.maddyhome.idea.vim.extension.ExtensionHandler | ||||||
| @@ -45,7 +45,8 @@ internal class ParagraphMotion : VimExtension { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     fun moveCaretToNextParagraph(editor: VimEditor, caret: Caret, count: Int): Int? { |     fun moveCaretToNextParagraph(editor: VimEditor, caret: Caret, count: Int): Int? { | ||||||
|       return injector.searchHelper.findNextParagraph(editor, caret.vim, count, true)?.let(editor::getLineEndForOffset) |       return injector.searchHelper.findNextParagraph(editor, caret.vim, count, true) | ||||||
|  |         ?.let { editor.normalizeOffset(it, true) } | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,321 +0,0 @@ | |||||||
| /* |  | ||||||
|  * 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.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.VimPlugin |  | ||||||
| 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.VimExtensionFacade.putKeyMapping |  | ||||||
| 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 javax.swing.Timer |  | ||||||
|  |  | ||||||
|  |  | ||||||
| private const val DEFAULT_HIGHLIGHT_DURATION_SNEAK = 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) { |  | ||||||
|       val timer = Timer(DEFAULT_HIGHLIGHT_DURATION_SNEAK) { |  | ||||||
|         if (editor?.isDisposed != true) { |  | ||||||
|           editor?.markupModel?.removeHighlighter(highlighter) |  | ||||||
|         } |  | ||||||
|       } |  | ||||||
|       timer.isRepeats = false |  | ||||||
|       timer.start() |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     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.putExtensionHandlerMapping( |  | ||||||
|     MappingMode.NXO, |  | ||||||
|     injector.parser.parseKeys(commandFromOriginalPlugin(keys)), |  | ||||||
|     owner, |  | ||||||
|     handler, |  | ||||||
|     false |  | ||||||
|   ) |  | ||||||
|  |  | ||||||
|   // This is a combination to meet the following requirements: |  | ||||||
|   //  - Now we should support mappings from sneak `Sneak_s` and mappings from the previous version of the plugin `(sneak-s)` |  | ||||||
|   //  - The shortcut should not be registered if any of these mappings is overridden in .ideavimrc |  | ||||||
|   //  - The shortcut should not be registered if some other shortcut for this key exists |  | ||||||
|   val fromKeys = injector.parser.parseKeys(keys) |  | ||||||
|   val filteredModes = MappingMode.NXO.filterNotTo(HashSet()) { |  | ||||||
|     VimPlugin.getKey().hasmapto(it, injector.parser.parseKeys(command(keys))) |  | ||||||
|   } |  | ||||||
|   val filteredModes2 = MappingMode.NXO.filterNotTo(HashSet()) { |  | ||||||
|     VimPlugin.getKey().hasmapto(it, injector.parser.parseKeys(commandFromOriginalPlugin(keys))) |  | ||||||
|   } |  | ||||||
|   val filteredFromModes = MappingMode.NXO.filterNotTo(HashSet()) { |  | ||||||
|     injector.keyGroup.hasmapfrom(it, fromKeys) |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   val doubleFiltered = MappingMode.NXO |  | ||||||
|     .filter { it in filteredModes2 && it in filteredModes && it in filteredFromModes } |  | ||||||
|     .toSet() |  | ||||||
|   putKeyMapping(doubleFiltered, fromKeys, owner, injector.parser.parseKeys(command(keys)), true) |  | ||||||
|   putKeyMapping( |  | ||||||
|     doubleFiltered, |  | ||||||
|     fromKeys, |  | ||||||
|     owner, |  | ||||||
|     injector.parser.parseKeys(commandFromOriginalPlugin(keys)), |  | ||||||
|     true |  | ||||||
|   ) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| private fun command(keys: String) = "<Plug>(sneak-$keys)" |  | ||||||
| private fun commandFromOriginalPlugin(keys: String) = "<Plug>Sneak_$keys" |  | ||||||
| @@ -1,30 +0,0 @@ | |||||||
| package com.maddyhome.idea.vim.extension.surround |  | ||||||
|  |  | ||||||
| import com.intellij.util.text.CharSequenceSubSequence |  | ||||||
|  |  | ||||||
| internal data class RepeatedCharSequence(val text: CharSequence, val count: Int) : CharSequence { |  | ||||||
|   override val length = text.length * count |  | ||||||
|  |  | ||||||
|   override fun get(index: Int): Char { |  | ||||||
|     if (index < 0 || index >= length) throw IndexOutOfBoundsException() |  | ||||||
|     return text[index % text.length] |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   override fun subSequence(startIndex: Int, endIndex: Int): CharSequence { |  | ||||||
|     return CharSequenceSubSequence(this, startIndex, endIndex) |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   override fun toString(): String { |  | ||||||
|     return text.repeat(count) |  | ||||||
|   } |  | ||||||
|    |  | ||||||
|   companion object { |  | ||||||
|     fun of(text: CharSequence, count: Int): CharSequence { |  | ||||||
|       return when (count) { |  | ||||||
|         0 -> "" |  | ||||||
|         1 -> text |  | ||||||
|         else -> RepeatedCharSequence(text, count) |  | ||||||
|       } |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
| } |  | ||||||
| @@ -8,7 +8,6 @@ | |||||||
| 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 | ||||||
| @@ -20,7 +19,10 @@ 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 | ||||||
| @@ -32,6 +34,7 @@ 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.helper.runWithEveryCaretAndRestore | import com.maddyhome.idea.vim.helper.runWithEveryCaretAndRestore | ||||||
| import com.maddyhome.idea.vim.key.OperatorFunction | import com.maddyhome.idea.vim.key.OperatorFunction | ||||||
| import com.maddyhome.idea.vim.newapi.IjVimCaret | import com.maddyhome.idea.vim.newapi.IjVimCaret | ||||||
| @@ -40,10 +43,6 @@ 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 | ||||||
| @@ -84,7 +83,7 @@ internal class VimSurroundExtension : VimExtension { | |||||||
|     override val isRepeatable = true |     override val isRepeatable = true | ||||||
|  |  | ||||||
|     override fun execute(editor: VimEditor, context: ExecutionContext, operatorArguments: OperatorArguments) { |     override fun execute(editor: VimEditor, context: ExecutionContext, operatorArguments: OperatorArguments) { | ||||||
|       setOperatorFunction(Operator(supportsMultipleCursors = false, count = 1)) // TODO |       setOperatorFunction(Operator(supportsMultipleCursors = false)) // TODO | ||||||
|       executeNormalWithoutMapping(injector.parser.parseKeys("g@"), editor.ij) |       executeNormalWithoutMapping(injector.parser.parseKeys("g@"), editor.ij) | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
| @@ -105,7 +104,7 @@ internal class VimSurroundExtension : VimExtension { | |||||||
|         val lastNonWhiteSpaceOffset = getLastNonWhitespaceCharacterOffset(editor.text(), lineStartOffset, lineEndOffset) |         val lastNonWhiteSpaceOffset = getLastNonWhitespaceCharacterOffset(editor.text(), lineStartOffset, lineEndOffset) | ||||||
|         if (lastNonWhiteSpaceOffset != null) { |         if (lastNonWhiteSpaceOffset != null) { | ||||||
|           val range = TextRange(lineStartOffset, lastNonWhiteSpaceOffset + 1) |           val range = TextRange(lineStartOffset, lastNonWhiteSpaceOffset + 1) | ||||||
|           performSurround(pair, range, it, count = operatorArguments.count1) |           performSurround(pair, range, it) | ||||||
|         } |         } | ||||||
| //        it.moveToOffset(lineStartOffset) | //        it.moveToOffset(lineStartOffset) | ||||||
|       } |       } | ||||||
| @@ -126,7 +125,7 @@ internal class VimSurroundExtension : VimExtension { | |||||||
|   private class VSurroundHandler : ExtensionHandler { |   private class VSurroundHandler : ExtensionHandler { | ||||||
|     override fun execute(editor: VimEditor, context: ExecutionContext, operatorArguments: OperatorArguments) { |     override fun execute(editor: VimEditor, context: ExecutionContext, operatorArguments: OperatorArguments) { | ||||||
|       // NB: Operator ignores SelectionType anyway |       // NB: Operator ignores SelectionType anyway | ||||||
|       if (!Operator(supportsMultipleCursors = true, count = operatorArguments.count1).apply(editor, context, editor.mode.selectionType)) { |       if (!Operator(supportsMultipleCursors = true).apply(editor, context, editor.mode.selectionType)) { | ||||||
|         return |         return | ||||||
|       } |       } | ||||||
|       runWriteAction { |       runWriteAction { | ||||||
| @@ -256,14 +255,13 @@ 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) } | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   private class Operator(private val supportsMultipleCursors: Boolean, private val count: Int) : OperatorFunction { |   private class Operator(private val supportsMultipleCursors: Boolean) : OperatorFunction { | ||||||
|     override fun apply(vimEditor: VimEditor, context: ExecutionContext, selectionType: SelectionType?): Boolean { |     override fun apply(vimEditor: VimEditor, context: ExecutionContext, selectionType: SelectionType?): Boolean { | ||||||
|       val editor = vimEditor.ij |       val editor = vimEditor.ij | ||||||
|       val c = getChar(editor) |       val c = getChar(editor) | ||||||
| @@ -275,11 +273,11 @@ internal class VimSurroundExtension : VimExtension { | |||||||
|         val change = VimPlugin.getChange() |         val change = VimPlugin.getChange() | ||||||
|         if (supportsMultipleCursors) { |         if (supportsMultipleCursors) { | ||||||
|           editor.runWithEveryCaretAndRestore { |           editor.runWithEveryCaretAndRestore { | ||||||
|             applyOnce(editor, change, pair, count) |             applyOnce(editor, change, pair) | ||||||
|           } |           } | ||||||
|         } |         } | ||||||
|         else { |         else { | ||||||
|           applyOnce(editor, change, pair, count) |           applyOnce(editor, change, pair) | ||||||
|           // Jump back to start |           // Jump back to start | ||||||
|           executeNormalWithoutMapping(injector.parser.parseKeys("`["), editor) |           executeNormalWithoutMapping(injector.parser.parseKeys("`["), editor) | ||||||
|         } |         } | ||||||
| @@ -287,15 +285,18 @@ internal class VimSurroundExtension : VimExtension { | |||||||
|       return true |       return true | ||||||
|     } |     } | ||||||
|      |      | ||||||
|     private fun applyOnce(editor: Editor, change: VimChangeGroup, pair: Pair<String, String>, count: Int) { |     private fun applyOnce(editor: Editor, change: VimChangeGroup, pair: Pair<String, String>) { | ||||||
|       // XXX: Will it work with line-wise or block-wise selections? |       // XXX: Will it work with line-wise or block-wise selections? | ||||||
|       val primaryCaret = editor.caretModel.primaryCaret |       val primaryCaret = editor.caretModel.primaryCaret | ||||||
|       val range = getSurroundRange(primaryCaret.vim) |       val range = getSurroundRange(primaryCaret.vim) | ||||||
|       if (range != null) { |       if (range != null) { | ||||||
|         val start = RepeatedCharSequence.of(pair.first, count) |         change.insertText(IjVimEditor(editor), IjVimCaret(primaryCaret), range.startOffset, pair.first) | ||||||
|         val end = RepeatedCharSequence.of(pair.second, count) |         change.insertText( | ||||||
|         change.insertText(IjVimEditor(editor), IjVimCaret(primaryCaret), range.startOffset, start) |           IjVimEditor(editor), | ||||||
|         change.insertText(IjVimEditor(editor), IjVimCaret(primaryCaret), range.endOffset + start.length, end) |           IjVimCaret(primaryCaret), | ||||||
|  |           range.endOffset + pair.first.length, | ||||||
|  |           pair.second | ||||||
|  |         ) | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -309,10 +310,8 @@ internal class VimSurroundExtension : VimExtension { | |||||||
|       } |       } | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
| } |  | ||||||
|  |  | ||||||
| private val LOG = logger<VimSurroundExtension>() |  | ||||||
|  |  | ||||||
|  |   companion object { | ||||||
|     private const val REGISTER = '"' |     private const val REGISTER = '"' | ||||||
|  |  | ||||||
|     private val tagNameAndAttributesCapturePattern = "(\\S+)([^>]*)>".toPattern() |     private val tagNameAndAttributesCapturePattern = "(\\S+)([^>]*)>".toPattern() | ||||||
| @@ -372,24 +371,22 @@ private fun getOrInputPair(c: Char, editor: Editor): Pair<String, String>? = whe | |||||||
|     private fun getChar(editor: Editor): Char { |     private fun getChar(editor: Editor): Char { | ||||||
|       val key = inputKeyStroke(editor) |       val key = inputKeyStroke(editor) | ||||||
|       val keyChar = key.keyChar |       val keyChar = key.keyChar | ||||||
|   val res = if (keyChar == KeyEvent.CHAR_UNDEFINED || keyChar.code == KeyEvent.VK_ESCAPE) { |       return if (keyChar == KeyEvent.CHAR_UNDEFINED || keyChar.code == KeyEvent.VK_ESCAPE) { | ||||||
|         0.toChar() |         0.toChar() | ||||||
|       } else { |       } else { | ||||||
|         keyChar |         keyChar | ||||||
|       } |       } | ||||||
|   LOG.trace("getChar: $res") |  | ||||||
|   return res |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
| private fun performSurround(pair: Pair<String, String>, range: TextRange, caret: VimCaret, count: Int, tagsOnNewLines: Boolean = false) { |     private fun performSurround(pair: Pair<String, String>, range: TextRange, caret: VimCaret, tagsOnNewLines: Boolean = false) { | ||||||
|       runWriteAction { |       runWriteAction { | ||||||
|         val editor = caret.editor |         val editor = caret.editor | ||||||
|         val change = VimPlugin.getChange() |         val change = VimPlugin.getChange() | ||||||
|     val leftSurround = RepeatedCharSequence.of(pair.first + if (tagsOnNewLines) "\n" else "", count) |         val leftSurround = pair.first + if (tagsOnNewLines) "\n" else "" | ||||||
|  |  | ||||||
|         val isEOF = range.endOffset == editor.text().length |         val isEOF = range.endOffset == editor.text().length | ||||||
|         val hasNewLine = editor.endsWithNewLine() |         val hasNewLine = editor.endsWithNewLine() | ||||||
|     val rightSurround = (if (tagsOnNewLines) { |         val rightSurround = if (tagsOnNewLines) { | ||||||
|           if (isEOF && !hasNewLine) { |           if (isEOF && !hasNewLine) { | ||||||
|             "\n" + pair.second |             "\n" + pair.second | ||||||
|           } else { |           } else { | ||||||
| @@ -397,13 +394,12 @@ private fun performSurround(pair: Pair<String, String>, range: TextRange, caret: | |||||||
|           } |           } | ||||||
|         } else { |         } else { | ||||||
|           pair.second |           pair.second | ||||||
|     }).let { RepeatedCharSequence.of(it, count) } |         } | ||||||
|  |  | ||||||
|         change.insertText(editor, caret, range.startOffset, leftSurround) |         change.insertText(editor, caret, range.startOffset, leftSurround) | ||||||
|         change.insertText(editor, caret, range.endOffset + leftSurround.length, rightSurround) |         change.insertText(editor, caret, range.endOffset + leftSurround.length, rightSurround) | ||||||
|     injector.markService.setChangeMarks( |         injector.markService.setChangeMarks(caret, TextRange(range.startOffset, range.endOffset + leftSurround.length + rightSurround.length)) | ||||||
|       caret, |       } | ||||||
|       TextRange(range.startOffset, range.endOffset + leftSurround.length + rightSurround.length) |     } | ||||||
|     ) |  | ||||||
|   } |   } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -16,11 +16,12 @@ import com.intellij.openapi.command.UndoConfirmationPolicy | |||||||
| import com.intellij.openapi.diagnostic.logger | import com.intellij.openapi.diagnostic.logger | ||||||
| import com.intellij.openapi.editor.Editor | import com.intellij.openapi.editor.Editor | ||||||
| import com.intellij.openapi.editor.LogicalPosition | import com.intellij.openapi.editor.LogicalPosition | ||||||
| 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.util.UserDataHolder | 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.text.StringUtil | import com.intellij.openapi.util.text.StringUtil | ||||||
| import com.intellij.psi.codeStyle.CodeStyleManager | import com.intellij.psi.codeStyle.CodeStyleManager | ||||||
| import com.intellij.psi.util.PsiUtilBase | import com.intellij.psi.util.PsiUtilBase | ||||||
| @@ -51,7 +52,6 @@ import com.maddyhome.idea.vim.group.visual.VimSelection | |||||||
| import com.maddyhome.idea.vim.group.visual.vimSetSystemSelectionSilently | import com.maddyhome.idea.vim.group.visual.vimSetSystemSelectionSilently | ||||||
| import com.maddyhome.idea.vim.handler.Motion | import com.maddyhome.idea.vim.handler.Motion | ||||||
| import com.maddyhome.idea.vim.handler.Motion.AbsoluteOffset | import com.maddyhome.idea.vim.handler.Motion.AbsoluteOffset | ||||||
| import com.maddyhome.idea.vim.handler.commandContinuation |  | ||||||
| import com.maddyhome.idea.vim.helper.CharacterHelper | import com.maddyhome.idea.vim.helper.CharacterHelper | ||||||
| import com.maddyhome.idea.vim.helper.CharacterHelper.changeCase | import com.maddyhome.idea.vim.helper.CharacterHelper.changeCase | ||||||
| import com.maddyhome.idea.vim.helper.CharacterHelper.charType | import com.maddyhome.idea.vim.helper.CharacterHelper.charType | ||||||
| @@ -62,28 +62,29 @@ 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 | ||||||
| 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.ij | import com.maddyhome.idea.vim.newapi.ij | ||||||
| import com.maddyhome.idea.vim.state.mode.Mode |  | ||||||
| import com.maddyhome.idea.vim.state.mode.Mode.VISUAL | import com.maddyhome.idea.vim.state.mode.Mode.VISUAL | ||||||
| import com.maddyhome.idea.vim.state.mode.SelectionType | import com.maddyhome.idea.vim.state.mode.SelectionType | ||||||
| import com.maddyhome.idea.vim.state.mode.mode |  | ||||||
| import com.maddyhome.idea.vim.vimscript.model.commands.SortOption | import com.maddyhome.idea.vim.vimscript.model.commands.SortOption | ||||||
| import org.jetbrains.annotations.TestOnly | import org.jetbrains.annotations.TestOnly | ||||||
| import java.math.BigInteger | import java.math.BigInteger | ||||||
| import java.util.* | import java.util.* | ||||||
| import java.util.function.Consumer | import java.util.function.Consumer | ||||||
| import kotlin.math.max | import kotlin.math.max | ||||||
|  | import kotlin.math.min | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * Provides all the insert/replace related functionality |  * Provides all the insert/replace related functionality | ||||||
|  */ |  */ | ||||||
| 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 | ||||||
| @@ -97,6 +98,10 @@ 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 | ||||||
| @@ -111,35 +116,6 @@ public class ChangeGroup : VimChangeGroupBase() { | |||||||
|     injector.scroll.scrollCaretIntoView(vimEditor) |     injector.scroll.scrollCaretIntoView(vimEditor) | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   /** |  | ||||||
|    * If this is REPLACE mode we need to turn off OVERWRITE before and then turn OVERWRITE back on after sending the |  | ||||||
|    * "ENTER" key. |  | ||||||
|    */ |  | ||||||
|   override fun processEnter( |  | ||||||
|     editor: VimEditor, |  | ||||||
|     caret: VimCaret, |  | ||||||
|     context: ExecutionContext, |  | ||||||
|   ) { |  | ||||||
|     if (editor.mode is Mode.REPLACE) { |  | ||||||
|       editor.insertMode = true |  | ||||||
|     } |  | ||||||
|     try { |  | ||||||
|       val continuation = (context.context as UserDataHolder).getUserData(commandContinuation) |  | ||||||
|       val ijEditor = editor.ij |  | ||||||
|       val ij = context.ij |  | ||||||
|       val ijCaret = caret.ij |  | ||||||
|       if (continuation != null) { |  | ||||||
|         continuation.execute(ijEditor, ijCaret, ij) |  | ||||||
|       } else { |  | ||||||
|         EnterAction().handler.execute(ijEditor, ijCaret, ij) |  | ||||||
|       } |  | ||||||
|     } finally { |  | ||||||
|       if (editor.mode is Mode.REPLACE) { |  | ||||||
|         editor.insertMode = false |  | ||||||
|       } |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   override fun getDeleteRangeAndType2( |   override fun getDeleteRangeAndType2( | ||||||
|     editor: VimEditor, |     editor: VimEditor, | ||||||
|     caret: VimCaret, |     caret: VimCaret, | ||||||
| @@ -394,7 +370,6 @@ public class ChangeGroup : VimChangeGroupBase() { | |||||||
|     context: ExecutionContext, |     context: ExecutionContext, | ||||||
|     range: TextRange, |     range: TextRange, | ||||||
|   ) { |   ) { | ||||||
|     val startPos = editor.offsetToBufferPosition(caret.offset.point) |  | ||||||
|     val startOffset = editor.getLineStartForOffset(range.startOffset) |     val startOffset = editor.getLineStartForOffset(range.startOffset) | ||||||
|     val endOffset = editor.getLineEndForOffset(range.endOffset) |     val endOffset = editor.getLineEndForOffset(range.endOffset) | ||||||
|     val ijEditor = (editor as IjVimEditor).editor |     val ijEditor = (editor as IjVimEditor).editor | ||||||
| @@ -419,7 +394,11 @@ public class ChangeGroup : VimChangeGroupBase() { | |||||||
|       } |       } | ||||||
|     } |     } | ||||||
|     val afterAction = { |     val afterAction = { | ||||||
|       caret.moveToOffset(injector.motion.moveCaretToLineStartSkipLeading(editor, startPos.line)) |       val firstLine = editor.offsetToBufferPosition( | ||||||
|  |         min(startOffset.toDouble(), endOffset.toDouble()).toInt() | ||||||
|  |       ).line | ||||||
|  |       val newOffset = injector.motion.moveCaretToLineStartSkipLeading(editor, firstLine) | ||||||
|  |       caret.moveToOffset(newOffset) | ||||||
|       restoreCursor(editor, caret, (caret as IjVimCaret).caret.logicalPosition.line) |       restoreCursor(editor, caret, (caret as IjVimCaret).caret.logicalPosition.line) | ||||||
|     } |     } | ||||||
|     if (project != null) { |     if (project != null) { | ||||||
| @@ -531,7 +510,7 @@ public class ChangeGroup : VimChangeGroupBase() { | |||||||
|         val soff = editor.getLineStartOffset(l) |         val soff = editor.getLineStartOffset(l) | ||||||
|         val eoff = editor.getLineEndOffset(l, true) |         val eoff = editor.getLineEndOffset(l, true) | ||||||
|         val woff = injector.motion.moveCaretToLineStartSkipLeading(editor, l) |         val woff = injector.motion.moveCaretToLineStartSkipLeading(editor, l) | ||||||
|         val col = editor.offsetToBufferPosition(woff).column |         val col = editor.offsetToVisualPosition(woff).column | ||||||
|         val limit = max(0.0, (col + dir * indentConfig.getTotalIndent(count)).toDouble()) |         val limit = max(0.0, (col + dir * indentConfig.getTotalIndent(count)).toDouble()) | ||||||
|           .toInt() |           .toInt() | ||||||
|         if (col > 0 || soff != eoff) { |         if (col > 0 || soff != eoff) { | ||||||
| @@ -632,6 +611,25 @@ 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") | ||||||
|   | |||||||
| @@ -89,17 +89,14 @@ public class FileGroup extends VimFileBase { | |||||||
|  |  | ||||||
|   @Nullable VirtualFile findFile(@NotNull String filename, @NotNull Project project) { |   @Nullable VirtualFile findFile(@NotNull String filename, @NotNull Project project) { | ||||||
|     VirtualFile found; |     VirtualFile found; | ||||||
|     // Vim supports both ~/ and ~\ (tested on Mac and Windows). On Windows, it supports forward- and back-slashes, but |     if (filename.length() > 2 && filename.charAt(0) == '~' && filename.charAt(1) == File.separatorChar) { | ||||||
|     // it only supports forward slash on Unix (tested on Mac) |       String homefile = filename.substring(2); | ||||||
|     // VFS works with both directory separators (tested on Mac and Windows) |  | ||||||
|     if (filename.startsWith("~/") || filename.startsWith("~\\")) { |  | ||||||
|       String relativePath = filename.substring(2); |  | ||||||
|       String dir = System.getProperty("user.home"); |       String dir = System.getProperty("user.home"); | ||||||
|       if (logger.isDebugEnabled()) { |       if (logger.isDebugEnabled()) { | ||||||
|         logger.debug("home dir file"); |         logger.debug("home dir file"); | ||||||
|         logger.debug("looking for " + relativePath + " in " + dir); |         logger.debug("looking for " + homefile + " in " + dir); | ||||||
|       } |       } | ||||||
|       found = LocalFileSystem.getInstance().refreshAndFindFileByIoFile(new File(dir, relativePath)); |       found = LocalFileSystem.getInstance().refreshAndFindFileByIoFile(new File(dir, homefile)); | ||||||
|     } |     } | ||||||
|     else { |     else { | ||||||
|       found = LocalFileSystem.getInstance().findFileByIoFile(new File(filename)); |       found = LocalFileSystem.getInstance().findFileByIoFile(new File(filename)); | ||||||
|   | |||||||
| @@ -29,15 +29,13 @@ 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 octopushandler: Boolean by optionProperty(IjOptions.octopushandler) | ||||||
|   public var oldundo: Boolean by optionProperty(IjOptions.oldundo) |   public var oldundo: Boolean by optionProperty(IjOptions.oldundo) | ||||||
|   public var unifyjumps: Boolean by optionProperty(IjOptions.unifyjumps) |   public var unifyjumps: Boolean by optionProperty(IjOptions.unifyjumps) | ||||||
|   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 useNewRegex: Boolean by optionProperty(IjOptions.useNewRegex) |  | ||||||
| } | } | ||||||
|  |  | ||||||
| /** | /** | ||||||
|   | |||||||
| @@ -80,14 +80,12 @@ public object IjOptions { | |||||||
|       "lookupkeys", |       "lookupkeys", | ||||||
|       "<Tab>,<Down>,<Up>,<Enter>,<Left>,<Right>,<C-Down>,<C-Up>,<PageUp>,<PageDown>,<C-J>,<C-Q>") |       "<Tab>,<Down>,<Up>,<Enter>,<Left>,<Right>,<C-Down>,<C-Up>,<PageUp>,<PageDown>,<C-J>,<C-Q>") | ||||||
|   ) |   ) | ||||||
|  |   public val octopushandler: ToggleOption = addOption(ToggleOption("octopushandler", GLOBAL, "octopushandler", false)) | ||||||
|  |   public val oldundo: ToggleOption = addOption(ToggleOption("oldundo", GLOBAL, "oldundo", true)) | ||||||
|   public val trackactionids: ToggleOption = addOption(ToggleOption("trackactionids", GLOBAL, "tai", false)) |   public val trackactionids: ToggleOption = addOption(ToggleOption("trackactionids", GLOBAL, "tai", false)) | ||||||
|   public val unifyjumps: ToggleOption = addOption(ToggleOption("unifyjumps", GLOBAL, "unifyjumps", true)) |   public val unifyjumps: ToggleOption = addOption(ToggleOption("unifyjumps", GLOBAL, "unifyjumps", true)) | ||||||
|  |   public val vimscriptFunctionAnnotation: ToggleOption = addOption(ToggleOption("vimscriptfunctionannotation", GLOBAL, "vimscriptfunctionannotation", true)) | ||||||
|   public val visualdelay: UnsignedNumberOption = addOption(UnsignedNumberOption("visualdelay", GLOBAL, "visualdelay", 100)) |   public val visualdelay: UnsignedNumberOption = addOption(UnsignedNumberOption("visualdelay", GLOBAL, "visualdelay", 100)) | ||||||
|   public val oldundo: ToggleOption = addOption(ToggleOption("oldundo", GLOBAL, "oldundo", 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 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,14 +24,16 @@ 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.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; | ||||||
| @@ -46,7 +48,6 @@ 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; | ||||||
|  |  | ||||||
| /** | /** | ||||||
| @@ -207,20 +208,46 @@ public class KeyGroup extends VimKeyGroupBase implements PersistentStateComponen | |||||||
|     registerRequiredShortcut(Collections.singletonList(keyStroke), owner); |     registerRequiredShortcut(Collections.singletonList(keyStroke), owner); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   public void registerCommandAction(@NotNull LazyVimCommand command) { |   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()) { |     if (ApplicationManager.getApplication().isUnitTestMode()) { | ||||||
|       initIdentityChecker(); |       initIdentityChecker(); | ||||||
|       for (List<KeyStroke> keys : command.getKeys()) { |       for (List<KeyStroke> keys : actionKeys) { | ||||||
|         checkCommand(command.getModes(), command, keys); |         checkCommand(actionModes, actionHolder.getInstance(), keys); | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     for (List<KeyStroke> keyStrokes : command.getKeys()) { |     for (List<KeyStroke> keyStrokes : actionKeys) { | ||||||
|       registerRequiredShortcut(keyStrokes, MappingOwner.IdeaVim.System.INSTANCE); |       registerRequiredShortcut(keyStrokes, MappingOwner.IdeaVim.System.INSTANCE); | ||||||
|  |  | ||||||
|       for (MappingMode mappingMode : command.getModes()) { |       for (MappingMode mappingMode : actionModes) { | ||||||
|         Node<LazyVimCommand> node = getKeyRoot(mappingMode); |         Node<VimActionsInitiator> node = getKeyRoot(mappingMode); | ||||||
|         NodesKt.addLeafs(node, keyStrokes, command); |         NodesKt.addLeafs(node, keyStrokes, actionHolder); | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
| @@ -228,14 +255,10 @@ public class KeyGroup extends VimKeyGroupBase implements PersistentStateComponen | |||||||
|   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) { | ||||||
|         if (!injector.getOptionGroup().getGlobalOptions().getOctopushandler() || |  | ||||||
|             !(key.getKeyCode() == KeyEvent.VK_ESCAPE && key.getModifiers() == 0) && |  | ||||||
|             !(key.getKeyCode() == KeyEvent.VK_ENTER && key.getModifiers() == 0)) { |  | ||||||
|         getRequiredShortcutKeys().add(new RequiredShortcut(key, owner)); |         getRequiredShortcutKeys().add(new RequiredShortcut(key, owner)); | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|   } |  | ||||||
|  |  | ||||||
|   public static @NotNull ShortcutSet toShortcutSet(@NotNull Collection<RequiredShortcut> requiredShortcuts) { |   public static @NotNull ShortcutSet toShortcutSet(@NotNull Collection<RequiredShortcut> requiredShortcuts) { | ||||||
|     final List<Shortcut> shortcuts = new ArrayList<>(); |     final List<Shortcut> shortcuts = new ArrayList<>(); | ||||||
|   | |||||||
| @@ -1,68 +0,0 @@ | |||||||
| package com.maddyhome.idea.vim.group |  | ||||||
|  |  | ||||||
| import com.intellij.codeInsight.daemon.ReferenceImporter |  | ||||||
| import com.intellij.openapi.actionSystem.CommonDataKeys |  | ||||||
| import com.intellij.openapi.actionSystem.DataContext |  | ||||||
| import com.intellij.openapi.application.ApplicationManager |  | ||||||
| import com.intellij.openapi.application.ReadAction |  | ||||||
| import com.intellij.openapi.command.WriteCommandAction |  | ||||||
| import com.intellij.openapi.editor.Editor |  | ||||||
| import com.intellij.openapi.fileEditor.FileDocumentManager |  | ||||||
| import com.intellij.openapi.progress.ProgressIndicator |  | ||||||
| import com.intellij.openapi.progress.ProgressManager |  | ||||||
| import com.intellij.openapi.progress.Task |  | ||||||
| import com.intellij.psi.PsiDocumentManager |  | ||||||
| import com.intellij.psi.PsiElement |  | ||||||
| import com.intellij.psi.PsiRecursiveElementWalkingVisitor |  | ||||||
| import java.util.function.BooleanSupplier |  | ||||||
|  |  | ||||||
| internal object MacroAutoImport { |  | ||||||
|   fun run(editor: Editor, dataContext: DataContext) { |  | ||||||
|     val project = CommonDataKeys.PROJECT.getData(dataContext) ?: return |  | ||||||
|     val file = PsiDocumentManager.getInstance(project).getPsiFile(editor.document) ?: return |  | ||||||
|  |  | ||||||
|     if (!FileDocumentManager.getInstance().requestWriting(editor.document, project)) { |  | ||||||
|       return |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     val importers = ReferenceImporter.EP_NAME.extensionList |  | ||||||
|     if (importers.isEmpty()) { |  | ||||||
|       return |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     ProgressManager.getInstance().run(object : Task.Backgroundable(project, "Auto import", true) { |  | ||||||
|       override fun run(indicator: ProgressIndicator) { |  | ||||||
|         val fixes = ReadAction.nonBlocking<List<BooleanSupplier>> { |  | ||||||
|           val fixes = mutableListOf<BooleanSupplier>() |  | ||||||
|  |  | ||||||
|           file.accept(object : PsiRecursiveElementWalkingVisitor() { |  | ||||||
|             override fun visitElement(element: PsiElement) { |  | ||||||
|               for (reference in element.references) { |  | ||||||
|                 if (reference.resolve() != null) { |  | ||||||
|                   continue |  | ||||||
|                 } |  | ||||||
|                 for (importer in importers) { |  | ||||||
|                   importer.computeAutoImportAtOffset(editor, file, element.textRange.startOffset, true) |  | ||||||
|                     ?.let(fixes::add) |  | ||||||
|                 } |  | ||||||
|               } |  | ||||||
|               super.visitElement(element) |  | ||||||
|             } |  | ||||||
|           }) |  | ||||||
|  |  | ||||||
|           return@nonBlocking fixes |  | ||||||
|         }.executeSynchronously() |  | ||||||
|  |  | ||||||
|         ApplicationManager.getApplication().invokeAndWait { |  | ||||||
|           WriteCommandAction.writeCommandAction(project) |  | ||||||
|             .withName("Auto Import") |  | ||||||
|             .withGroupId("IdeaVimAutoImportAfterMacro") |  | ||||||
|             .shouldRecordActionForActiveDocument(true) |  | ||||||
|             .run<RuntimeException> { |  | ||||||
|               fixes.forEach { it.asBoolean } |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|       } |  | ||||||
|     }) |  | ||||||
|   } |  | ||||||
| } |  | ||||||
| @@ -7,8 +7,6 @@ | |||||||
|  */ |  */ | ||||||
| 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 | ||||||
| @@ -21,7 +19,6 @@ import com.maddyhome.idea.vim.api.injector | |||||||
| import com.maddyhome.idea.vim.helper.MessageHelper.message | import com.maddyhome.idea.vim.helper.MessageHelper.message | ||||||
| import com.maddyhome.idea.vim.macro.VimMacroBase | import com.maddyhome.idea.vim.macro.VimMacroBase | ||||||
| import com.maddyhome.idea.vim.newapi.IjVimEditor | import com.maddyhome.idea.vim.newapi.IjVimEditor | ||||||
| import com.maddyhome.idea.vim.newapi.ij |  | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * Used to handle playback of macros |  * Used to handle playback of macros | ||||||
| @@ -64,10 +61,8 @@ 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@{ | ||||||
|         try { |  | ||||||
|         // Handle one keystroke then queue up the next key |         // Handle one keystroke then queue up the next key | ||||||
|         for (i in 0 until total) { |         for (i in 0 until total) { | ||||||
|             try { |  | ||||||
|           myPotemkinProgress.fraction = (i + 1).toDouble() / total |           myPotemkinProgress.fraction = (i + 1).toDouble() / total | ||||||
|           while (keyStack.hasStroke()) { |           while (keyStack.hasStroke()) { | ||||||
|             val key = keyStack.feedStroke() |             val key = keyStack.feedStroke() | ||||||
| @@ -76,25 +71,13 @@ internal class MacroGroup : VimMacroBase() { | |||||||
|             } catch (e: ProcessCanceledException) { |             } catch (e: ProcessCanceledException) { | ||||||
|               return@runnable |               return@runnable | ||||||
|             } |             } | ||||||
|                 ProgressManager.getInstance().executeNonCancelableSection { |             ProgressManager.getInstance().executeNonCancelableSection { getInstance().handleKey(editor, key, context) } | ||||||
|                   // 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 |             if (injector.messages.isError()) return@runnable | ||||||
|           } |           } | ||||||
|             } finally { |  | ||||||
|           keyStack.resetFirst() |           keyStack.resetFirst() | ||||||
|         } |         } | ||||||
|           } |  | ||||||
|         } finally { |  | ||||||
|         keyStack.removeFirst() |         keyStack.removeFirst() | ||||||
|       } |       } | ||||||
|         if (!isInternalMacro) { |  | ||||||
|           MacroAutoImport.run(editor.ij, context.ij) |  | ||||||
|         } |  | ||||||
|       } |  | ||||||
|  |  | ||||||
|       if (isInternalMacro) { |       if (isInternalMacro) { | ||||||
|         runnable() |         runnable() | ||||||
|   | |||||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user