mirror of
				https://github.com/chylex/IntelliJ-IdeaVim.git
				synced 2025-10-25 03:23:43 +02:00 
			
		
		
		
	Compare commits
	
		
			15 Commits
		
	
	
		
			ea703bce69
			...
			customized
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 4ca05ba6df | |||
| 114ac40990 | |||
| 2829109eb7 | |||
| f1323139b4 | |||
| df4aa59310 | |||
| 7a3bb5b2d7 | |||
| c4b05957fc | |||
| db83b89931 | |||
| ce27c3e5ba | |||
| 31358bc983 | |||
| f7e1c9c837 | |||
| c46109caa3 | |||
| 9f7ca83306 | |||
| b5761f20d2 | |||
| 305c6d2bf9 | 
							
								
								
									
										1
									
								
								.gitattributes
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								.gitattributes
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1 @@ | |||||||
|  | * text=auto eol=lf | ||||||
							
								
								
									
										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: | ||||||
|   | |||||||
							
								
								
									
										25
									
								
								.github/workflows/runUiTests.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										25
									
								
								.github/workflows/runUiTests.yml
									
									
									
									
										vendored
									
									
								
							| @@ -5,23 +5,20 @@ 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: 11 |           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 | ||||||
| @@ -29,7 +26,7 @@ jobs: | |||||||
|           mkdir -p build/reports |           mkdir -p build/reports | ||||||
|           gradle :runIdeForUiTests > build/reports/idea.log & |           gradle :runIdeForUiTests > build/reports/idea.log & | ||||||
|       - name: Wait for Idea started |       - name: Wait for Idea started | ||||||
|         uses: jtalk/url-health-check-action@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 | ||||||
| @@ -37,19 +34,15 @@ jobs: | |||||||
|       - name: Tests |       - name: Tests | ||||||
|         run: gradle :testUi |         run: gradle :testUi | ||||||
|       - name: Move video |       - name: Move video | ||||||
|         if: always() |         if: ${{ failure() }} | ||||||
|         run: mv 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 | ||||||
|   | |||||||
							
								
								
									
										3
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										3
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -23,9 +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 | ||||||
| # 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 | ||||||
|   | |||||||
							
								
								
									
										1
									
								
								.idea/codeStyles/Project.xml
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										1
									
								
								.idea/codeStyles/Project.xml
									
									
									
										generated
									
									
									
								
							| @@ -6,6 +6,7 @@ | |||||||
|         <option name="CONTINUATION_INDENT_SIZE" value="4" /> |         <option name="CONTINUATION_INDENT_SIZE" value="4" /> | ||||||
|       </value> |       </value> | ||||||
|     </option> |     </option> | ||||||
|  |     <option name="LINE_SEPARATOR" value="
" /> | ||||||
|     <JavaCodeStyleSettings> |     <JavaCodeStyleSettings> | ||||||
|       <option name="FIELD_NAME_PREFIX" value="my" /> |       <option name="FIELD_NAME_PREFIX" value="my" /> | ||||||
|       <option name="STATIC_FIELD_NAME_PREFIX" value="our" /> |       <option name="STATIC_FIELD_NAME_PREFIX" value="our" /> | ||||||
|   | |||||||
							
								
								
									
										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" | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										6
									
								
								.teamcity/_Self/Project.kt
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										6
									
								
								.teamcity/_Self/Project.kt
									
									
									
									
										vendored
									
									
								
							| @@ -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/ReleasePlugin.kt
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								.teamcity/_Self/buildTypes/ReleasePlugin.kt
									
									
									
									
										vendored
									
									
								
							| @@ -118,7 +118,6 @@ sealed class ReleasePlugin(private val releaseType: String) : IdeaVimBuildType({ | |||||||
|       then |       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 --tags | ||||||
|         git push origin --force |         git push origin --force | ||||||
|       fi |       fi | ||||||
|   | |||||||
							
								
								
									
										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" |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
							
								
								
									
										20
									
								
								.teamcity/patches/buildTypes/ReleaseMinor.kts
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										20
									
								
								.teamcity/patches/buildTypes/ReleaseMinor.kts
									
									
									
									
										vendored
									
									
								
							| @@ -1,20 +0,0 @@ | |||||||
| package patches.buildTypes |  | ||||||
|  |  | ||||||
| import jetbrains.buildServer.configs.kotlin.v2019_2.* |  | ||||||
| import jetbrains.buildServer.configs.kotlin.v2019_2.ui.* |  | ||||||
|  |  | ||||||
| /* |  | ||||||
| This patch script was generated by TeamCity on settings change in UI. |  | ||||||
| To apply the patch, change the buildType with id = 'ReleaseMinor' |  | ||||||
| accordingly, and delete the patch script. |  | ||||||
| */ |  | ||||||
| changeBuildType(RelativeId("ReleaseMinor")) { |  | ||||||
|     params { |  | ||||||
|         expect { |  | ||||||
|             password("env.ORG_GRADLE_PROJECT_youtrackToken", "credentialsJSON:3cd3e867-282c-451f-b958-bc95d56a8450", display = ParameterDisplay.HIDDEN) |  | ||||||
|         } |  | ||||||
|         update { |  | ||||||
|             password("env.ORG_GRADLE_PROJECT_youtrackToken", "credentialsJSON:7bc0eb3a-b86a-4ebd-b622-d4ef12d7e1d3", display = ParameterDisplay.HIDDEN) |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
							
								
								
									
										17
									
								
								.teamcity/patches/projects/_Self.kts
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										17
									
								
								.teamcity/patches/projects/_Self.kts
									
									
									
									
										vendored
									
									
								
							| @@ -1,17 +0,0 @@ | |||||||
| package patches.projects |  | ||||||
|  |  | ||||||
| import jetbrains.buildServer.configs.kotlin.v2019_2.* |  | ||||||
| import jetbrains.buildServer.configs.kotlin.v2019_2.Project |  | ||||||
| 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 root project |  | ||||||
| accordingly, and delete the patch script. |  | ||||||
| */ |  | ||||||
| changeProject(DslContext.projectId) { |  | ||||||
|     check(description == "Vim engine for IDEs based on the IntelliJ platform") { |  | ||||||
|         "Unexpected description: '$description'" |  | ||||||
|     } |  | ||||||
|     description = "Vim engine for JetBrains IDEs" |  | ||||||
| } |  | ||||||
| @@ -483,14 +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 |  | ||||||
|  |  | ||||||
| Previous contributors: | Previous contributors: | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										62
									
								
								CHANGES.md
									
									
									
									
									
								
							
							
						
						
									
										62
									
								
								CHANGES.md
									
									
									
									
									
								
							| @@ -25,73 +25,11 @@ usual beta standards. | |||||||
|  |  | ||||||
| ## To Be Released | ## To Be Released | ||||||
|  |  | ||||||
| ### 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 |  | ||||||
|  |  | ||||||
| ### 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`. | ||||||
|   | |||||||
| @@ -8,11 +8,8 @@ | |||||||
|  |  | ||||||
| plugins { | plugins { | ||||||
|   kotlin("jvm") |   kotlin("jvm") | ||||||
|   kotlin("plugin.serialization") version "1.8.21" |  | ||||||
| } | } | ||||||
|  |  | ||||||
| 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.16") |   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,14 +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 kotlinx.serialization.Serializable |  | ||||||
|  |  | ||||||
| @Serializable |  | ||||||
| data class CommandBean(val keys: String, val `class`: String, val modes: String) |  | ||||||
| @@ -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 | ||||||
| @@ -49,14 +49,14 @@ buildscript { | |||||||
|         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") | ||||||
| @@ -69,7 +69,7 @@ plugins { | |||||||
|     kotlin("jvm") version "1.8.21" |     kotlin("jvm") version "1.8.21" | ||||||
|     application |     application | ||||||
|  |  | ||||||
|     id("org.jetbrains.intellij") version "1.16.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 |     // ktlint linter - read more: https://github.com/JLLeitschuh/ktlint-gradle | ||||||
| @@ -82,10 +82,9 @@ plugins { | |||||||
| } | } | ||||||
|  |  | ||||||
| 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 { | ||||||
| @@ -116,7 +115,7 @@ repositories { | |||||||
|  |  | ||||||
| dependencies { | dependencies { | ||||||
|     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") | ||||||
|  |  | ||||||
|     // 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") | ||||||
| @@ -126,24 +125,24 @@ dependencies { | |||||||
|     testImplementation("org.jetbrains.kotlin:kotlin-test:$kotlinVersion") |     testImplementation("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("com.intellij.remoterobot:remote-robot:$remoteRobotVersion") |     testImplementation("com.intellij.remoterobot:remote-robot:$remoteRobotVersion") | ||||||
|     testImplementation("com.intellij.remoterobot:remote-fixtures:$remoteRobotVersion") |     testImplementation("com.intellij.remoterobot:remote-fixtures:$remoteRobotVersion") | ||||||
|     testImplementation("com.automation-remarks:video-recorder-junit5:2.0") |     testImplementation("com.automation-remarks:video-recorder-junit:2.0") | ||||||
|     runtimeOnly("org.antlr:antlr4-runtime:$antlrVersion") |     runtimeOnly("org.antlr:antlr4-runtime:$antlrVersion") | ||||||
|     antlr("org.antlr:antlr4:$antlrVersion") |     antlr("org.antlr:antlr4:$antlrVersion") | ||||||
|  |  | ||||||
|     api(project(":vim-engine")) |     api(project(":vim-engine")) | ||||||
|  |  | ||||||
|     ksp(project(":annotation-processors")) |     ksp(project(":annotation-processors")) | ||||||
|     implementation(project(":annotation-processors")) |     compileOnly(project(":annotation-processors")) | ||||||
|  |  | ||||||
|     testApi("com.squareup.okhttp3:okhttp:4.12.0") |     testApi("com.squareup.okhttp3:okhttp:4.11.0") | ||||||
|  |  | ||||||
|     testImplementation("org.junit.jupiter:junit-jupiter-api:5.10.1") |     testImplementation("org.junit.jupiter:junit-jupiter-api:5.10.0") | ||||||
|     testImplementation("org.junit.jupiter:junit-jupiter-engine:5.10.1") |     testImplementation("org.junit.jupiter:junit-jupiter-engine:5.10.0") | ||||||
|     testImplementation("org.junit.jupiter:junit-jupiter-params:5.10.1") |     testImplementation("org.junit.jupiter:junit-jupiter-params:5.10.0") | ||||||
| } | } | ||||||
|  |  | ||||||
| configurations { | configurations { | ||||||
| @@ -184,14 +183,6 @@ tasks { | |||||||
|         include("**/*test.class") |         include("**/*test.class") | ||||||
|         include("**/*Tests.class") |         include("**/*Tests.class") | ||||||
|         exclude("**/ParserTest.class") |         exclude("**/ParserTest.class") | ||||||
|  |  | ||||||
|         // Set teamcity env variable locally to run additional tests for leaks. |  | ||||||
|         // By default, this test runs on TC only, but this test doesn't take a lot of time, |  | ||||||
|         //   so we can turn it on for local development |  | ||||||
|         if (environment["TEAMCITY_VERSION"] == null) { |  | ||||||
|             println("Set env TEAMCITY_VERSION to X") |  | ||||||
|             environment("TEAMCITY_VERSION" to "X") |  | ||||||
|         } |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     val testWithNeovim by getting(Test::class) { |     val testWithNeovim by getting(Test::class) { | ||||||
| @@ -302,7 +293,6 @@ tasks { | |||||||
|         systemProperty("ide.mac.message.dialogs.as.sheets", "false") |         systemProperty("ide.mac.message.dialogs.as.sheets", "false") | ||||||
|         systemProperty("jb.privacy.policy.text", "<!--999.999-->") |         systemProperty("jb.privacy.policy.text", "<!--999.999-->") | ||||||
|         systemProperty("jb.consents.confirmation.enabled", "false") |         systemProperty("jb.consents.confirmation.enabled", "false") | ||||||
|         systemProperty("ide.show.tips.on.startup.default.value", "false") |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     runPluginVerifier { |     runPluginVerifier { | ||||||
| @@ -353,7 +343,7 @@ tasks { | |||||||
|     val pluginVersion = version |     val pluginVersion = version | ||||||
|     // Don't forget to update plugin.xml |     // Don't forget to update plugin.xml | ||||||
|     patchPluginXml { |     patchPluginXml { | ||||||
|         sinceBuild.set("233.11799.30") |         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( | ||||||
| @@ -533,12 +523,10 @@ tasks.register("releaseActions") { | |||||||
|         if (tickets.isNotEmpty()) { |         if (tickets.isNotEmpty()) { | ||||||
|             println("Updating statuses for tickets: $tickets") |             println("Updating statuses for tickets: $tickets") | ||||||
|             setYoutrackStatus(tickets, "Fixed") |             setYoutrackStatus(tickets, "Fixed") | ||||||
|             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 { | ||||||
|   | |||||||
| @@ -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]] | ||||||
|   | |||||||
| @@ -77,7 +77,7 @@ Original plugin: [NERDTree](https://github.com/preservim/nerdtree). | |||||||
|     |     | ||||||
| ### Instructions | ### Instructions | ||||||
|     |     | ||||||
| [See here](NERDTree-support.md). | [[See here|NERDTree-support]]. | ||||||
|  |  | ||||||
| </details> | </details> | ||||||
|  |  | ||||||
| @@ -396,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: | ||||||
|  |  | ||||||
| ``` | ``` | ||||||
|   | |||||||
| @@ -8,25 +8,21 @@ | |||||||
|  |  | ||||||
| # suppress inspection "UnusedProperty" for whole file | # suppress inspection "UnusedProperty" for whole file | ||||||
|  |  | ||||||
| ideaVersion=2023.3.2 | ideaVersion=2023.2 | ||||||
| downloadIdeaSources=true | downloadIdeaSources=true | ||||||
| instrumentPluginCode=true | instrumentPluginCode=true | ||||||
| version=SNAPSHOT | version=chylex-19 | ||||||
| javaVersion=17 | javaVersion=17 | ||||||
| remoteRobotVersion=0.11.21 | remoteRobotVersion=0.11.17 | ||||||
| antlrVersion=4.10.1 | antlrVersion=4.10.1 | ||||||
|  |  | ||||||
|  | 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.8.21 | ||||||
| publishToken=token | publishToken=token | ||||||
| publishChannels=eap | publishChannels=eap | ||||||
|  |  | ||||||
| # Kotlinx serialization also uses some version of kotlin stdlib under the hood. However, |  | ||||||
| #   we exclude this version from the dependency and use our own version of kotlin that is specified above |  | ||||||
| kotlinxSerializationVersion=1.5.1 |  | ||||||
|  |  | ||||||
| slackUrl= | slackUrl= | ||||||
| youtrackToken= | youtrackToken= | ||||||
|  |  | ||||||
|   | |||||||
| @@ -20,17 +20,17 @@ repositories { | |||||||
| } | } | ||||||
|  |  | ||||||
| dependencies { | dependencies { | ||||||
|   compileOnly("org.jetbrains.kotlin:kotlin-stdlib:1.9.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") | ||||||
| } | } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -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() | ||||||
|   | |||||||
| @@ -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() |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -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,100 +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.intellij.openapi.application.ApplicationManager |  | ||||||
| import com.intellij.openapi.extensions.ExtensionPointName |  | ||||||
| import com.maddyhome.idea.vim.action.EngineCommandProvider |  | ||||||
| import com.maddyhome.idea.vim.action.IntellijCommandProvider |  | ||||||
| import com.maddyhome.idea.vim.api.injector |  | ||||||
| import com.maddyhome.idea.vim.handler.ActionBeanClass |  | ||||||
| import com.maddyhome.idea.vim.handler.EditorActionHandlerBase |  | ||||||
| import com.maddyhome.idea.vim.key.MappingOwner |  | ||||||
| import com.maddyhome.idea.vim.newapi.IjVimActionsInitiator |  | ||||||
| import com.maddyhome.idea.vim.newapi.globalIjOptions |  | ||||||
| import java.awt.event.KeyEvent |  | ||||||
| import javax.swing.KeyStroke |  | ||||||
|  |  | ||||||
| public object RegisterActions { |  | ||||||
|   @Deprecated("Please use @CommandOrMotion annotation instead") |  | ||||||
|   internal val VIM_ACTIONS_EP: ExtensionPointName<ActionBeanClass> = ExtensionPointName.create("IdeaVIM.vimAction") |  | ||||||
|  |  | ||||||
|   /** |  | ||||||
|    * Register all the key/action mappings for the plugin. |  | ||||||
|    */ |  | ||||||
|   @JvmStatic |  | ||||||
|   public fun registerActions() { |  | ||||||
|     registerVimCommandActions() |  | ||||||
|     if (!injector.globalIjOptions().commandOrMotionAnnotation) { |  | ||||||
|       registerEmptyShortcuts() |  | ||||||
|       registerEpListener() |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   @Deprecated("Moving to annotations approach instead of xml") |  | ||||||
|   private fun registerEpListener() { |  | ||||||
|     // IdeaVim doesn't support contribution to VIM_ACTIONS_EP extension point, so technically we can skip this update, |  | ||||||
|     //   but let's support dynamic plugins in a more classic way and reload actions on every EP change. |  | ||||||
|     VIM_ACTIONS_EP.addChangeListener({ |  | ||||||
|       unregisterActions() |  | ||||||
|       registerActions() |  | ||||||
|     }, VimPlugin.getInstance()) |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   public fun findAction(id: String): EditorActionHandlerBase? { |  | ||||||
|     if (injector.globalIjOptions().commandOrMotionAnnotation) { |  | ||||||
|       val commandBean = EngineCommandProvider.getCommands().firstOrNull { it.actionId == id } |  | ||||||
|         ?: IntellijCommandProvider.getCommands().firstOrNull { it.actionId == id } ?: return null |  | ||||||
|       return commandBean.instance |  | ||||||
|     } else { |  | ||||||
|       return VIM_ACTIONS_EP.getExtensionList(ApplicationManager.getApplication()).stream() |  | ||||||
|         .filter { vimActionBean: ActionBeanClass -> vimActionBean.actionId == id } |  | ||||||
|         .findFirst().map { obj: ActionBeanClass -> obj.instance } |  | ||||||
|         .orElse(null) |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   public fun findActionOrDie(id: String): EditorActionHandlerBase { |  | ||||||
|     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() |  | ||||||
|     if (injector.globalIjOptions().commandOrMotionAnnotation) { |  | ||||||
|       EngineCommandProvider.getCommands().forEach { parser.registerCommandAction(it) } |  | ||||||
|       IntellijCommandProvider.getCommands().forEach { parser.registerCommandAction(it) } |  | ||||||
|     } else { |  | ||||||
|       VIM_ACTIONS_EP.getExtensionList(ApplicationManager.getApplication()).stream().map { bean: ActionBeanClass? -> |  | ||||||
|         IjVimActionsInitiator( |  | ||||||
|           bean!! |  | ||||||
|         ) |  | ||||||
|       } |  | ||||||
|         .forEach { actionHolder: IjVimActionsInitiator? -> |  | ||||||
|           parser.registerCommandAction( |  | ||||||
|             actionHolder!! |  | ||||||
|           ) |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   // todo do we really need this? |  | ||||||
|   private fun registerEmptyShortcuts() { |  | ||||||
|     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) |  | ||||||
|   } |  | ||||||
| } |  | ||||||
| @@ -219,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; | ||||||
|  |  | ||||||
| @@ -236,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() { | ||||||
| @@ -274,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); | ||||||
| @@ -317,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 | ||||||
| @@ -345,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 | ||||||
| @@ -28,6 +32,7 @@ import com.maddyhome.idea.vim.api.globalOptions | |||||||
| import com.maddyhome.idea.vim.api.injector | import com.maddyhome.idea.vim.api.injector | ||||||
| import com.maddyhome.idea.vim.group.IjOptionConstants | import com.maddyhome.idea.vim.group.IjOptionConstants | ||||||
| import com.maddyhome.idea.vim.group.IjOptions | import com.maddyhome.idea.vim.group.IjOptions | ||||||
|  | import com.maddyhome.idea.vim.handler.enableOctopus | ||||||
| import com.maddyhome.idea.vim.handler.isOctopusEnabled | import com.maddyhome.idea.vim.handler.isOctopusEnabled | ||||||
| import com.maddyhome.idea.vim.helper.EditorHelper | import com.maddyhome.idea.vim.helper.EditorHelper | ||||||
| import com.maddyhome.idea.vim.helper.HandlerInjector | import com.maddyhome.idea.vim.helper.HandlerInjector | ||||||
| @@ -54,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") | ||||||
| @@ -98,29 +95,29 @@ 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 (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) { | ||||||
|         return ActionEnableStatus.no("IdeaVim is disabled in this place", LogLevel.INFO) |         return ActionEnableStatus.no("IdeaVim is disabled in this place", LogLevel.INFO) | ||||||
| @@ -167,6 +164,14 @@ 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) | ||||||
|       } |       } | ||||||
|  |  | ||||||
|  |       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 | ||||||
| @@ -232,9 +237,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 | ||||||
| @@ -242,9 +247,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 | ||||||
|       } |       } | ||||||
| @@ -277,7 +282,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)) | ||||||
| @@ -368,12 +373,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) |  | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
| @@ -386,5 +389,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, | ||||||
|   | |||||||
| @@ -8,8 +8,6 @@ | |||||||
|  |  | ||||||
| package com.maddyhome.idea.vim.action.editor | package com.maddyhome.idea.vim.action.editor | ||||||
|  |  | ||||||
| import com.intellij.vim.annotations.CommandOrMotion |  | ||||||
| import com.intellij.vim.annotations.Mode |  | ||||||
| import com.intellij.openapi.actionSystem.IdeActions | import com.intellij.openapi.actionSystem.IdeActions | ||||||
| import com.maddyhome.idea.vim.action.ComplicatedKeysAction | import com.maddyhome.idea.vim.action.ComplicatedKeysAction | ||||||
| import com.maddyhome.idea.vim.api.ExecutionContext | import com.maddyhome.idea.vim.api.ExecutionContext | ||||||
| @@ -25,7 +23,6 @@ import java.awt.event.KeyEvent | |||||||
| import java.util.* | import java.util.* | ||||||
| import javax.swing.KeyStroke | 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), ComplicatedKeysAction { | ||||||
|   override val keyStrokesSet: Set<List<KeyStroke>> = setOf( |   override val keyStrokesSet: Set<List<KeyStroke>> = setOf( | ||||||
|     listOf(KeyStroke.getKeyStroke(KeyEvent.VK_H, KeyEvent.CTRL_DOWN_MASK)), |     listOf(KeyStroke.getKeyStroke(KeyEvent.VK_H, KeyEvent.CTRL_DOWN_MASK)), | ||||||
| @@ -34,7 +31,6 @@ internal class VimEditorBackSpace : IdeActionHandler(IdeActions.ACTION_EDITOR_BA | |||||||
|   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), ComplicatedKeysAction { | ||||||
|   override val keyStrokesSet: Set<List<KeyStroke>> = setOf( |   override val keyStrokesSet: Set<List<KeyStroke>> = setOf( | ||||||
|     listOf(KeyStroke.getKeyStroke(KeyEvent.VK_DELETE, 0)), |     listOf(KeyStroke.getKeyStroke(KeyEvent.VK_DELETE, 0)), | ||||||
| @@ -43,7 +39,6 @@ internal class VimEditorDelete : IdeActionHandler(IdeActions.ACTION_EDITOR_DELET | |||||||
|   override val flags: EnumSet<CommandFlags> = enumSetOf(CommandFlags.FLAG_SAVE_STROKE) |   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), ComplicatedKeysAction { | ||||||
|   override val keyStrokesSet: Set<List<KeyStroke>> = setOf( |   override val keyStrokesSet: Set<List<KeyStroke>> = setOf( | ||||||
|     listOf(KeyStroke.getKeyStroke(KeyEvent.VK_DOWN, 0)), |     listOf(KeyStroke.getKeyStroke(KeyEvent.VK_DOWN, 0)), | ||||||
| @@ -53,7 +48,6 @@ internal class VimEditorDown : IdeActionHandler(IdeActions.ACTION_EDITOR_MOVE_CA | |||||||
|   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), ComplicatedKeysAction { | ||||||
|   override val keyStrokesSet: Set<List<KeyStroke>> = setOf( |   override val keyStrokesSet: Set<List<KeyStroke>> = setOf( | ||||||
|     listOf(KeyStroke.getKeyStroke(KeyEvent.VK_I, KeyEvent.CTRL_DOWN_MASK)), |     listOf(KeyStroke.getKeyStroke(KeyEvent.VK_I, KeyEvent.CTRL_DOWN_MASK)), | ||||||
| @@ -63,7 +57,6 @@ internal class VimEditorTab : IdeActionHandler(IdeActions.ACTION_EDITOR_TAB), Co | |||||||
|   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), ComplicatedKeysAction { | ||||||
|   override val keyStrokesSet: Set<List<KeyStroke>> = setOf( |   override val keyStrokesSet: Set<List<KeyStroke>> = setOf( | ||||||
|     listOf(KeyStroke.getKeyStroke(KeyEvent.VK_UP, 0)), |     listOf(KeyStroke.getKeyStroke(KeyEvent.VK_UP, 0)), | ||||||
| @@ -73,7 +66,6 @@ internal class VimEditorUp : IdeActionHandler(IdeActions.ACTION_EDITOR_MOVE_CARE | |||||||
|   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,8 +7,6 @@ | |||||||
|  */ |  */ | ||||||
| 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.action.ComplicatedKeysAction | ||||||
| import com.maddyhome.idea.vim.api.ExecutionContext | import com.maddyhome.idea.vim.api.ExecutionContext | ||||||
| @@ -25,7 +23,6 @@ import javax.swing.KeyStroke | |||||||
|  * |  * | ||||||
|  * The mapping for this action means that the ex command is executed as a write action |  * The mapping for this action means that the ex command is executed as a write action | ||||||
|  */ |  */ | ||||||
| @CommandOrMotion(keys = ["<CR>", "<C-M>", "<C-J>"], modes = [Mode.CMD_LINE]) |  | ||||||
| public class ProcessExEntryAction : VimActionHandler.SingleExecution(), ComplicatedKeysAction { | public class ProcessExEntryAction : VimActionHandler.SingleExecution(), ComplicatedKeysAction { | ||||||
|   override val keyStrokesSet: Set<List<KeyStroke>> = |   override val keyStrokesSet: Set<List<KeyStroke>> = | ||||||
|     parseKeysSet("<CR>", "<C-M>", 0x0a.toChar().toString(), 0x0d.toChar().toString()) |     parseKeysSet("<CR>", "<C-M>", 0x0a.toChar().toString(), 0x0d.toChar().toString()) | ||||||
|   | |||||||
| @@ -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()) | ||||||
|   | |||||||
| @@ -82,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) | ||||||
|   | |||||||
| @@ -231,7 +231,7 @@ private object FileTypePatterns { | |||||||
|     } else if (fileTypeName == "CMakeLists.txt" || fileName == "CMakeLists") { |     } else if (fileTypeName == "CMakeLists.txt" || fileName == "CMakeLists") { | ||||||
|       this.cMakePatterns |       this.cMakePatterns | ||||||
|     } else { |     } else { | ||||||
|       return null |       this.htmlPatterns | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -0,0 +1,30 @@ | |||||||
|  | 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,11 +8,11 @@ | |||||||
| 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 | ||||||
| import com.maddyhome.idea.vim.api.VimCaret | import com.maddyhome.idea.vim.api.VimCaret | ||||||
|  | import com.maddyhome.idea.vim.api.VimChangeGroup | ||||||
| import com.maddyhome.idea.vim.api.VimEditor | import com.maddyhome.idea.vim.api.VimEditor | ||||||
| import com.maddyhome.idea.vim.api.endsWithNewLine | import com.maddyhome.idea.vim.api.endsWithNewLine | ||||||
| import com.maddyhome.idea.vim.api.getLeadingCharacterOffset | import com.maddyhome.idea.vim.api.getLeadingCharacterOffset | ||||||
| @@ -31,7 +31,10 @@ 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.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.IjVimEditor | ||||||
| import com.maddyhome.idea.vim.newapi.ij | import com.maddyhome.idea.vim.newapi.ij | ||||||
| import com.maddyhome.idea.vim.newapi.vim | import com.maddyhome.idea.vim.newapi.vim | ||||||
| import com.maddyhome.idea.vim.options.helpers.ClipboardOptionHelper | import com.maddyhome.idea.vim.options.helpers.ClipboardOptionHelper | ||||||
| @@ -80,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()) |       setOperatorFunction(Operator(supportsMultipleCursors = false, count = 1)) // TODO | ||||||
|       executeNormalWithoutMapping(injector.parser.parseKeys("g@"), editor.ij) |       executeNormalWithoutMapping(injector.parser.parseKeys("g@"), editor.ij) | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
| @@ -101,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) |           performSurround(pair, range, it, count = operatorArguments.count1) | ||||||
|         } |         } | ||||||
| //        it.moveToOffset(lineStartOffset) | //        it.moveToOffset(lineStartOffset) | ||||||
|       } |       } | ||||||
| @@ -121,15 +124,13 @@ 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) { | ||||||
|       val selectionStart = editor.ij.caretModel.primaryCaret.selectionStart |  | ||||||
|       // NB: Operator ignores SelectionType anyway |       // NB: Operator ignores SelectionType anyway | ||||||
|       if (!Operator().apply(editor, context, editor.mode.selectionType)) { |       if (!Operator(supportsMultipleCursors = true, count = operatorArguments.count1).apply(editor, context, editor.mode.selectionType)) { | ||||||
|         return |         return | ||||||
|       } |       } | ||||||
|       runWriteAction { |       runWriteAction { | ||||||
|         // Leave visual mode |         // Leave visual mode | ||||||
|         executeNormalWithoutMapping(injector.parser.parseKeys("<Esc>"), editor.ij) |         executeNormalWithoutMapping(injector.parser.parseKeys("<Esc>"), editor.ij) | ||||||
|         editor.ij.caretModel.moveToOffset(selectionStart) |  | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
| @@ -150,6 +151,10 @@ internal class VimSurroundExtension : VimExtension { | |||||||
|  |  | ||||||
|     companion object { |     companion object { | ||||||
|       fun change(editor: VimEditor, context: ExecutionContext, charFrom: Char, newSurround: Pair<String, String>?) { |       fun change(editor: VimEditor, context: ExecutionContext, charFrom: Char, newSurround: Pair<String, String>?) { | ||||||
|  |         editor.ij.runWithEveryCaretAndRestore { changeAtCaret(editor, context, charFrom, newSurround) } | ||||||
|  |       } | ||||||
|  |        | ||||||
|  |       fun changeAtCaret(editor: VimEditor, context: ExecutionContext, charFrom: Char, newSurround: Pair<String, String>?) { | ||||||
|         // Save old register values for carets |         // Save old register values for carets | ||||||
|         val surroundings = editor.sortedCarets() |         val surroundings = editor.sortedCarets() | ||||||
|           .map { |           .map { | ||||||
| @@ -250,28 +255,48 @@ 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 : OperatorFunction { |   private class Operator(private val supportsMultipleCursors: Boolean, private val count: Int) : OperatorFunction { | ||||||
|     override fun apply(editor: VimEditor, context: ExecutionContext, selectionType: SelectionType?): Boolean { |     override fun apply(vimEditor: VimEditor, context: ExecutionContext, selectionType: SelectionType?): Boolean { | ||||||
|       val ijEditor = editor.ij |       val editor = vimEditor.ij | ||||||
|       val c = getChar(ijEditor) |       val c = getChar(editor) | ||||||
|       if (c.code == 0) return true |       if (c.code == 0) return true | ||||||
|  |  | ||||||
|       val pair = getOrInputPair(c, ijEditor) ?: return false |       val pair = getOrInputPair(c, editor) ?: return false | ||||||
|       // XXX: Will it work with line-wise or block-wise selections? |  | ||||||
|       val range = getSurroundRange(editor.currentCaret()) ?: return false |       runWriteAction { | ||||||
|       performSurround(pair, range, editor.currentCaret(), selectionType == SelectionType.LINE_WISE) |         val change = VimPlugin.getChange() | ||||||
|  |         if (supportsMultipleCursors) { | ||||||
|  |           editor.runWithEveryCaretAndRestore { | ||||||
|  |             applyOnce(editor, change, pair, count) | ||||||
|  |           } | ||||||
|  |         } | ||||||
|  |         else { | ||||||
|  |           applyOnce(editor, change, pair, count) | ||||||
|           // Jump back to start |           // Jump back to start | ||||||
|       executeNormalWithoutMapping(injector.parser.parseKeys("`["), ijEditor) |           executeNormalWithoutMapping(injector.parser.parseKeys("`["), editor) | ||||||
|  |         } | ||||||
|  |       } | ||||||
|       return true |       return true | ||||||
|     } |     } | ||||||
|      |      | ||||||
|  |     private fun applyOnce(editor: Editor, change: VimChangeGroup, pair: Pair<String, String>, count: Int) { | ||||||
|  |       // XXX: Will it work with line-wise or block-wise selections? | ||||||
|  |       val primaryCaret = editor.caretModel.primaryCaret | ||||||
|  |       val range = getSurroundRange(primaryCaret.vim) | ||||||
|  |       if (range != null) { | ||||||
|  |         val start = RepeatedCharSequence.of(pair.first, count) | ||||||
|  |         val end = RepeatedCharSequence.of(pair.second, count) | ||||||
|  |         change.insertText(IjVimEditor(editor), IjVimCaret(primaryCaret), range.startOffset, start) | ||||||
|  |         change.insertText(IjVimEditor(editor), IjVimCaret(primaryCaret), range.endOffset + start.length, end) | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |  | ||||||
|     private fun getSurroundRange(caret: VimCaret): TextRange? { |     private fun getSurroundRange(caret: VimCaret): TextRange? { | ||||||
|       val editor = caret.editor |       val editor = caret.editor | ||||||
|       val ijEditor = editor.ij |       val ijEditor = editor.ij | ||||||
| @@ -282,15 +307,13 @@ 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() |     private val SURROUND_PAIRS = mapOf( | ||||||
|  |  | ||||||
| private val SURROUND_PAIRS = mapOf( |  | ||||||
|       'b' to ("(" to ")"), |       'b' to ("(" to ")"), | ||||||
|       '(' to ("( " to " )"), |       '(' to ("( " to " )"), | ||||||
|       ')' to ("(" to ")"), |       ')' to ("(" to ")"), | ||||||
| @@ -303,18 +326,18 @@ private val SURROUND_PAIRS = mapOf( | |||||||
|       'a' to ("<" to ">"), |       'a' to ("<" to ">"), | ||||||
|       '>' to ("<" to ">"), |       '>' to ("<" to ">"), | ||||||
|       's' to (" " to ""), |       's' to (" " to ""), | ||||||
| ) |     ) | ||||||
|  |  | ||||||
| private fun getSurroundPair(c: Char): Pair<String, String>? = if (c in SURROUND_PAIRS) { |     private fun getSurroundPair(c: Char): Pair<String, String>? = if (c in SURROUND_PAIRS) { | ||||||
|       SURROUND_PAIRS[c] |       SURROUND_PAIRS[c] | ||||||
| } else if (!c.isLetter()) { |     } else if (!c.isLetter()) { | ||||||
|       val s = c.toString() |       val s = c.toString() | ||||||
|       s to s |       s to s | ||||||
| } else { |     } else { | ||||||
|       null |       null | ||||||
| } |     } | ||||||
|  |  | ||||||
| private fun inputTagPair(editor: Editor): Pair<String, String>? { |     private fun inputTagPair(editor: Editor): Pair<String, String>? { | ||||||
|       val tagInput = inputString(editor, "<", '>') |       val tagInput = inputString(editor, "<", '>') | ||||||
|       val matcher = tagNameAndAttributesCapturePattern.matcher(tagInput) |       val matcher = tagNameAndAttributesCapturePattern.matcher(tagInput) | ||||||
|       return if (matcher.find()) { |       return if (matcher.find()) { | ||||||
| @@ -324,45 +347,43 @@ private fun inputTagPair(editor: Editor): Pair<String, String>? { | |||||||
|       } else { |       } else { | ||||||
|         null |         null | ||||||
|       } |       } | ||||||
| } |     } | ||||||
|  |  | ||||||
| private fun inputFunctionName( |     private fun inputFunctionName( | ||||||
|       editor: Editor, |       editor: Editor, | ||||||
|       withInternalSpaces: Boolean, |       withInternalSpaces: Boolean, | ||||||
| ): Pair<String, String>? { |     ): Pair<String, String>? { | ||||||
|       val functionNameInput = inputString(editor, "function: ", null) |       val functionNameInput = inputString(editor, "function: ", null) | ||||||
|       if (functionNameInput.isEmpty()) return null |       if (functionNameInput.isEmpty()) return null | ||||||
|       return if (withInternalSpaces) "$functionNameInput( " to " )" else "$functionNameInput(" to ")" |       return if (withInternalSpaces) "$functionNameInput( " to " )" else "$functionNameInput(" to ")" | ||||||
| } |     } | ||||||
|  |  | ||||||
| private fun getOrInputPair(c: Char, editor: Editor): Pair<String, String>? = when (c) { |     private fun getOrInputPair(c: Char, editor: Editor): Pair<String, String>? = when (c) { | ||||||
|       '<', 't' -> inputTagPair(editor) |       '<', 't' -> inputTagPair(editor) | ||||||
|       'f' -> inputFunctionName(editor, false) |       'f' -> inputFunctionName(editor, false) | ||||||
|       'F' -> inputFunctionName(editor, true) |       'F' -> inputFunctionName(editor, true) | ||||||
|       else -> getSurroundPair(c) |       else -> getSurroundPair(c) | ||||||
| } |     } | ||||||
|  |  | ||||||
| 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, tagsOnNewLines: Boolean = false) { |     private fun performSurround(pair: Pair<String, String>, range: TextRange, caret: VimCaret, count: Int, tagsOnNewLines: Boolean = false) { | ||||||
|       runWriteAction { |       runWriteAction { | ||||||
|         val editor = caret.editor |         val editor = caret.editor | ||||||
|         val change = VimPlugin.getChange() |         val change = VimPlugin.getChange() | ||||||
|     val leftSurround = pair.first + if (tagsOnNewLines) "\n" else "" |         val leftSurround = RepeatedCharSequence.of(pair.first + if (tagsOnNewLines) "\n" else "", count) | ||||||
|  |  | ||||||
|         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 { | ||||||
| @@ -370,13 +391,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,16 +62,15 @@ 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 | ||||||
| @@ -85,6 +84,7 @@ import kotlin.math.min | |||||||
|  */ |  */ | ||||||
| public class ChangeGroup : VimChangeGroupBase() { | public class ChangeGroup : VimChangeGroupBase() { | ||||||
|   private val insertListeners = ContainerUtil.createLockFreeCopyOnWriteList<VimInsertListener>() |   private val insertListeners = ContainerUtil.createLockFreeCopyOnWriteList<VimInsertListener>() | ||||||
|  |   private var lastShownTime = 0L | ||||||
|   private val listener: EditorMouseListener = object : EditorMouseListener { |   private val listener: EditorMouseListener = object : EditorMouseListener { | ||||||
|     override fun mouseClicked(event: EditorMouseEvent) { |     override fun mouseClicked(event: EditorMouseEvent) { | ||||||
|       val editor = event.editor |       val editor = event.editor | ||||||
| @@ -98,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 | ||||||
| @@ -112,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, | ||||||
| @@ -636,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) |  | ||||||
|   public var colorfulmodewidget: Boolean by optionProperty(IjOptions.colorfulmodewidget) |  | ||||||
|  |  | ||||||
|   // 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) |  | ||||||
| } | } | ||||||
|  |  | ||||||
| /** | /** | ||||||
|   | |||||||
| @@ -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", false, 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 colorfulmodewidget: ToggleOption = addOption(ToggleOption("colorfulmodewidget", GLOBAL, "colorfulmodewidget", false, 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> | ||||||
|   | |||||||
| @@ -26,7 +26,6 @@ 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.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; | ||||||
| @@ -209,25 +208,6 @@ public class KeyGroup extends VimKeyGroupBase implements PersistentStateComponen | |||||||
|     registerRequiredShortcut(Collections.singletonList(keyStroke), owner); |     registerRequiredShortcut(Collections.singletonList(keyStroke), owner); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   public void registerCommandAction(@NotNull LazyVimCommand command) { |  | ||||||
|     if (ApplicationManager.getApplication().isUnitTestMode()) { |  | ||||||
|       initIdentityChecker(); |  | ||||||
|       for (List<KeyStroke> keys : command.getKeys()) { |  | ||||||
|         checkCommand(command.getModes(), command, keys); |  | ||||||
|       } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     for (List<KeyStroke> keyStrokes : command.getKeys()) { |  | ||||||
|       registerRequiredShortcut(keyStrokes, MappingOwner.IdeaVim.System.INSTANCE); |  | ||||||
|  |  | ||||||
|       for (MappingMode mappingMode : command.getModes()) { |  | ||||||
|         Node<VimActionsInitiator> node = getKeyRoot(mappingMode); |  | ||||||
|         NodesKt.addLeafs(node, keyStrokes, command); |  | ||||||
|       } |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   @Deprecated |  | ||||||
|   public void registerCommandAction(@NotNull VimActionsInitiator actionHolder) { |   public void registerCommandAction(@NotNull VimActionsInitiator actionHolder) { | ||||||
|     IjVimActionsInitiator holder = (IjVimActionsInitiator)actionHolder; |     IjVimActionsInitiator holder = (IjVimActionsInitiator)actionHolder; | ||||||
|  |  | ||||||
| @@ -274,9 +254,7 @@ 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) { | ||||||
|           !(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)); | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -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 | ||||||
| @@ -63,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() | ||||||
| @@ -75,22 +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) { |       if (isInternalMacro) { | ||||||
|         runnable() |         runnable() | ||||||
|   | |||||||
| @@ -28,7 +28,7 @@ import java.util.Set; | |||||||
| @Deprecated | @Deprecated | ||||||
| @ApiStatus.ScheduledForRemoval(inVersion = "2.3") | @ApiStatus.ScheduledForRemoval(inVersion = "2.3") | ||||||
| public class MarkGroup { | public class MarkGroup { | ||||||
|   public List<Jump> jumps = VimInjectorKt.injector.getJumpService().getJumps(""); |   public List<Jump> jumps = VimInjectorKt.injector.getJumpService().getJumps(); | ||||||
|  |  | ||||||
|   public void saveJumpLocation(@NotNull Editor editor) { |   public void saveJumpLocation(@NotNull Editor editor) { | ||||||
|     VimInjectorKt.injector.getJumpService().saveJumpLocation(new IjVimEditor(editor)); |     VimInjectorKt.injector.getJumpService().saveJumpLocation(new IjVimEditor(editor)); | ||||||
| @@ -54,7 +54,7 @@ public class MarkGroup { | |||||||
|  |  | ||||||
|   @Nullable |   @Nullable | ||||||
|   public Jump getJump(int count) { |   public Jump getJump(int count) { | ||||||
|     return VimInjectorKt.injector.getJumpService().getJump("", count); |     return VimInjectorKt.injector.getJumpService().getJump(count); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   @Nullable |   @Nullable | ||||||
| @@ -115,7 +115,7 @@ public class MarkGroup { | |||||||
|   } |   } | ||||||
|  |  | ||||||
|   public int getJumpSpot() { |   public int getJumpSpot() { | ||||||
|     return VimInjectorKt.injector.getJumpService().getJumpSpot(""); |     return VimInjectorKt.injector.getJumpService().getJumpSpot(); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   public void updateMarkFromDelete(@Nullable VimEditor editor, |   public void updateMarkFromDelete(@Nullable VimEditor editor, | ||||||
| @@ -133,6 +133,6 @@ public class MarkGroup { | |||||||
|   } |   } | ||||||
|  |  | ||||||
|   public void dropLastJump() { |   public void dropLastJump() { | ||||||
|     VimInjectorKt.injector.getJumpService().dropLastJump(""); |     VimInjectorKt.injector.getJumpService().dropLastJump(); | ||||||
|   } |   } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -33,8 +33,6 @@ import com.maddyhome.idea.vim.api.VimEditor | |||||||
| import com.maddyhome.idea.vim.api.VimMotionGroupBase | import com.maddyhome.idea.vim.api.VimMotionGroupBase | ||||||
| import com.maddyhome.idea.vim.api.addJump | import com.maddyhome.idea.vim.api.addJump | ||||||
| import com.maddyhome.idea.vim.api.anyNonWhitespace | import com.maddyhome.idea.vim.api.anyNonWhitespace | ||||||
| import com.maddyhome.idea.vim.api.getJump |  | ||||||
| import com.maddyhome.idea.vim.api.getJumpSpot |  | ||||||
| import com.maddyhome.idea.vim.api.getLeadingCharacterOffset | import com.maddyhome.idea.vim.api.getLeadingCharacterOffset | ||||||
| import com.maddyhome.idea.vim.api.getVisualLineCount | import com.maddyhome.idea.vim.api.getVisualLineCount | ||||||
| import com.maddyhome.idea.vim.api.injector | import com.maddyhome.idea.vim.api.injector | ||||||
| @@ -48,7 +46,9 @@ import com.maddyhome.idea.vim.api.options | |||||||
| import com.maddyhome.idea.vim.api.visualLineToBufferLine | import com.maddyhome.idea.vim.api.visualLineToBufferLine | ||||||
| import com.maddyhome.idea.vim.command.Argument | import com.maddyhome.idea.vim.command.Argument | ||||||
| import com.maddyhome.idea.vim.command.MotionType | import com.maddyhome.idea.vim.command.MotionType | ||||||
|  | import com.maddyhome.idea.vim.state.mode.Mode | ||||||
| import com.maddyhome.idea.vim.command.OperatorArguments | import com.maddyhome.idea.vim.command.OperatorArguments | ||||||
|  | import com.maddyhome.idea.vim.state.VimStateMachine | ||||||
| import com.maddyhome.idea.vim.common.TextRange | import com.maddyhome.idea.vim.common.TextRange | ||||||
| import com.maddyhome.idea.vim.ex.ExOutputModel | import com.maddyhome.idea.vim.ex.ExOutputModel | ||||||
| import com.maddyhome.idea.vim.handler.Motion | import com.maddyhome.idea.vim.handler.Motion | ||||||
| @@ -72,8 +72,6 @@ import com.maddyhome.idea.vim.newapi.IjVimCaret | |||||||
| import com.maddyhome.idea.vim.newapi.IjVimEditor | import com.maddyhome.idea.vim.newapi.IjVimEditor | ||||||
| import com.maddyhome.idea.vim.newapi.ij | import com.maddyhome.idea.vim.newapi.ij | ||||||
| import com.maddyhome.idea.vim.newapi.vim | import com.maddyhome.idea.vim.newapi.vim | ||||||
| import com.maddyhome.idea.vim.state.VimStateMachine |  | ||||||
| import com.maddyhome.idea.vim.state.mode.Mode |  | ||||||
| import com.maddyhome.idea.vim.ui.ex.ExEntryPanel | import com.maddyhome.idea.vim.ui.ex.ExEntryPanel | ||||||
| import org.jetbrains.annotations.Range | import org.jetbrains.annotations.Range | ||||||
| import java.io.File | import java.io.File | ||||||
| @@ -165,8 +163,8 @@ internal class MotionGroup : VimMotionGroupBase() { | |||||||
|  |  | ||||||
|   override fun moveCaretToJump(editor: VimEditor, caret: ImmutableVimCaret, count: Int): Motion { |   override fun moveCaretToJump(editor: VimEditor, caret: ImmutableVimCaret, count: Int): Motion { | ||||||
|     val jumpService = injector.jumpService |     val jumpService = injector.jumpService | ||||||
|     val spot = jumpService.getJumpSpot(editor) |     val spot = jumpService.getJumpSpot() | ||||||
|     val (line, col, fileName) = jumpService.getJump(editor, count) ?: return Motion.Error |     val (line, col, fileName) = jumpService.getJump(count) ?: return Motion.Error | ||||||
|     val vf = EditorHelper.getVirtualFile(editor.ij) ?: return Motion.Error |     val vf = EditorHelper.getVirtualFile(editor.ij) ?: return Motion.Error | ||||||
|     val lp = BufferPosition(line, col, false) |     val lp = BufferPosition(line, col, false) | ||||||
|     val lpNative = LogicalPosition(line, col, false) |     val lpNative = LogicalPosition(line, col, false) | ||||||
| @@ -461,7 +459,6 @@ internal class MotionGroup : VimMotionGroupBase() { | |||||||
|       val fileEditor = event.oldEditor |       val fileEditor = event.oldEditor | ||||||
|       if (fileEditor is TextEditor) { |       if (fileEditor is TextEditor) { | ||||||
|         val editor = fileEditor.editor |         val editor = fileEditor.editor | ||||||
|         if (!editor.isDisposed) { |  | ||||||
|         ExOutputModel.getInstance(editor).clear() |         ExOutputModel.getInstance(editor).clear() | ||||||
|         editor.vim.let { vimEditor -> |         editor.vim.let { vimEditor -> | ||||||
|           if (VimStateMachine.getInstance(vimEditor).mode is Mode.VISUAL) { |           if (VimStateMachine.getInstance(vimEditor).mode is Mode.VISUAL) { | ||||||
| @@ -472,5 +469,4 @@ internal class MotionGroup : VimMotionGroupBase() { | |||||||
|       } |       } | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|   } |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -21,11 +21,8 @@ import com.intellij.openapi.actionSystem.ActionUpdateThread | |||||||
| import com.intellij.openapi.actionSystem.AnAction | import com.intellij.openapi.actionSystem.AnAction | ||||||
| import com.intellij.openapi.actionSystem.AnActionEvent | import com.intellij.openapi.actionSystem.AnActionEvent | ||||||
| import com.intellij.openapi.actionSystem.KeyboardShortcut | import com.intellij.openapi.actionSystem.KeyboardShortcut | ||||||
| import com.intellij.openapi.diagnostic.logger |  | ||||||
| import com.intellij.openapi.ide.CopyPasteManager | import com.intellij.openapi.ide.CopyPasteManager | ||||||
| import com.intellij.openapi.keymap.KeymapUtil | import com.intellij.openapi.keymap.KeymapUtil | ||||||
| import com.intellij.openapi.keymap.ex.KeymapManagerEx |  | ||||||
| import com.intellij.openapi.keymap.impl.ui.KeymapPanel |  | ||||||
| import com.intellij.openapi.options.ShowSettingsUtil | import com.intellij.openapi.options.ShowSettingsUtil | ||||||
| import com.intellij.openapi.project.DumbAwareAction | import com.intellij.openapi.project.DumbAwareAction | ||||||
| import com.intellij.openapi.project.Project | import com.intellij.openapi.project.Project | ||||||
| @@ -35,7 +32,6 @@ import com.maddyhome.idea.vim.VimPlugin | |||||||
| import com.maddyhome.idea.vim.api.VimEditor | import com.maddyhome.idea.vim.api.VimEditor | ||||||
| import com.maddyhome.idea.vim.api.globalOptions | import com.maddyhome.idea.vim.api.globalOptions | ||||||
| import com.maddyhome.idea.vim.api.injector | import com.maddyhome.idea.vim.api.injector | ||||||
| import com.maddyhome.idea.vim.handler.KeyMapIssue |  | ||||||
| import com.maddyhome.idea.vim.helper.MessageHelper | import com.maddyhome.idea.vim.helper.MessageHelper | ||||||
| import com.maddyhome.idea.vim.key.ShortcutOwner | import com.maddyhome.idea.vim.key.ShortcutOwner | ||||||
| import com.maddyhome.idea.vim.key.ShortcutOwnerInfo | import com.maddyhome.idea.vim.key.ShortcutOwnerInfo | ||||||
| @@ -184,77 +180,6 @@ internal class NotificationService(private val project: Project?) { | |||||||
|     ActionIdNotifier.notifyActionId(id, project) |     ActionIdNotifier.notifyActionId(id, project) | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   fun notifyKeymapIssues(issues: ArrayList<KeyMapIssue>) { |  | ||||||
|     val keymapManager = KeymapManagerEx.getInstanceEx() |  | ||||||
|     val keymap = keymapManager.activeKeymap |  | ||||||
|     val message = buildString { |  | ||||||
|       appendLine("Current IDE keymap (${keymap.name}) has issues:<br/>") |  | ||||||
|       issues.forEach { |  | ||||||
|         when (it) { |  | ||||||
|           is KeyMapIssue.AddShortcut -> { |  | ||||||
|             appendLine("- ${it.key} key is not assigned to the ${it.action} action.<br/>") |  | ||||||
|           } |  | ||||||
|           is KeyMapIssue.RemoveShortcut -> { |  | ||||||
|             appendLine("- ${it.shortcut} key is incorrectly assigned to the ${it.action} action.<br/>") |  | ||||||
|           } |  | ||||||
|         } |  | ||||||
|       } |  | ||||||
|     } |  | ||||||
|     val notification = IDEAVIM_STICKY_GROUP.createNotification( |  | ||||||
|       IDEAVIM_NOTIFICATION_TITLE, |  | ||||||
|       message, |  | ||||||
|       NotificationType.ERROR, |  | ||||||
|     ) |  | ||||||
|     notification.subtitle = "IDE keymap misconfigured" |  | ||||||
|     notification.addAction(object : DumbAwareAction("Fix Keymap") { |  | ||||||
|       override fun actionPerformed(e: AnActionEvent) { |  | ||||||
|         issues.forEach { |  | ||||||
|           when (it) { |  | ||||||
|             is KeyMapIssue.AddShortcut -> { |  | ||||||
|               keymap.addShortcut(it.actionId, KeyboardShortcut(it.keyStroke, null)) |  | ||||||
|             } |  | ||||||
|  |  | ||||||
|             is KeyMapIssue.RemoveShortcut -> { |  | ||||||
|               keymap.removeShortcut(it.actionId, it.shortcut) |  | ||||||
|             } |  | ||||||
|           } |  | ||||||
|         } |  | ||||||
|         LOG.info("Shortcuts updated $issues") |  | ||||||
|         notification.expire() |  | ||||||
|         requiredShortcutsAssigned() |  | ||||||
|       } |  | ||||||
|     }) |  | ||||||
|     notification.addAction(object : DumbAwareAction("Open Keymap Settings") { |  | ||||||
|       override fun actionPerformed(e: AnActionEvent) { |  | ||||||
|         ShowSettingsUtil.getInstance().showSettingsDialog(e.project, KeymapPanel::class.java) |  | ||||||
|         notification.hideBalloon() |  | ||||||
|       } |  | ||||||
|     }) |  | ||||||
|     notification.addAction(object : DumbAwareAction("Ignore") { |  | ||||||
|       override fun actionPerformed(e: AnActionEvent) { |  | ||||||
|         LOG.info("Ignored to update shortcuts $issues") |  | ||||||
|         notification.hideBalloon() |  | ||||||
|       } |  | ||||||
|     }) |  | ||||||
|     notification.notify(project) |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   private fun requiredShortcutsAssigned() { |  | ||||||
|     val notification = Notification( |  | ||||||
|       IDEAVIM_NOTIFICATION_ID, |  | ||||||
|       IDEAVIM_NOTIFICATION_TITLE, |  | ||||||
|       "Keymap fixed", |  | ||||||
|       NotificationType.INFORMATION, |  | ||||||
|     ) |  | ||||||
|     notification.addAction(object : DumbAwareAction("Open Keymap Settings") { |  | ||||||
|       override fun actionPerformed(e: AnActionEvent) { |  | ||||||
|         ShowSettingsUtil.getInstance().showSettingsDialog(e.project, KeymapPanel::class.java) |  | ||||||
|         notification.hideBalloon() |  | ||||||
|       } |  | ||||||
|     }) |  | ||||||
|     notification.notify(project) |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   object ActionIdNotifier { |   object ActionIdNotifier { | ||||||
|     private var notification: Notification? = null |     private var notification: Notification? = null | ||||||
|     private const val NO_ID = "<i>Cannot detect action id</i>" |     private const val NO_ID = "<i>Cannot detect action id</i>" | ||||||
| @@ -389,8 +314,6 @@ internal class NotificationService(private val project: Project?) { | |||||||
|     const val IDEAVIM_NOTIFICATION_TITLE = "IdeaVim" |     const val IDEAVIM_NOTIFICATION_TITLE = "IdeaVim" | ||||||
|     const val ideajoinExamplesUrl = "https://jb.gg/f9zji9" |     const val ideajoinExamplesUrl = "https://jb.gg/f9zji9" | ||||||
|  |  | ||||||
|     private val LOG = logger<NotificationService>() |  | ||||||
|  |  | ||||||
|     private fun createIdeaVimRcManually(message: String, project: Project?) { |     private fun createIdeaVimRcManually(message: String, project: Project?) { | ||||||
|       val notification = |       val notification = | ||||||
|         Notification(IDEAVIM_NOTIFICATION_ID, IDEAVIM_NOTIFICATION_TITLE, message, NotificationType.WARNING) |         Notification(IDEAVIM_NOTIFICATION_ID, IDEAVIM_NOTIFICATION_TITLE, message, NotificationType.WARNING) | ||||||
|   | |||||||
| @@ -40,6 +40,10 @@ internal class OptionGroup : VimOptionGroupBase(), IjVimOptionGroup { | |||||||
|   override fun getGlobalIjOptions() = GlobalIjOptions(OptionAccessScope.GLOBAL(null)) |   override fun getGlobalIjOptions() = GlobalIjOptions(OptionAccessScope.GLOBAL(null)) | ||||||
|   override fun getEffectiveIjOptions(editor: VimEditor) = EffectiveIjOptions(OptionAccessScope.EFFECTIVE(editor)) |   override fun getEffectiveIjOptions(editor: VimEditor) = EffectiveIjOptions(OptionAccessScope.EFFECTIVE(editor)) | ||||||
|  |  | ||||||
|  |   private fun updateFallbackWindow(fallbackWindow: VimEditor, targetEditor: VimEditor) { | ||||||
|  |     copyPerWindowGlobalValues(fallbackWindow, targetEditor) | ||||||
|  |   } | ||||||
|  |  | ||||||
|   companion object { |   companion object { | ||||||
|     fun fileEditorManagerSelectionChangedCallback(event: FileEditorManagerEvent) { |     fun fileEditorManagerSelectionChangedCallback(event: FileEditorManagerEvent) { | ||||||
|       // Vim only has one window, and it's not possible to close it. This means that editing a new file will always |       // Vim only has one window, and it's not possible to close it. This means that editing a new file will always | ||||||
| @@ -54,8 +58,6 @@ internal class OptionGroup : VimOptionGroupBase(), IjVimOptionGroup { | |||||||
|       // Unfortunately, we can't reliably know if a closing editor is the selected editor. Instead, we rely on selection |       // Unfortunately, we can't reliably know if a closing editor is the selected editor. Instead, we rely on selection | ||||||
|       // change events. If an editor is losing selection and there is no new selection, we can assume this means that |       // change events. If an editor is losing selection and there is no new selection, we can assume this means that | ||||||
|       // the last editor has been closed, and use the closed editor to update the fallback window |       // the last editor has been closed, and use the closed editor to update the fallback window | ||||||
|       // |  | ||||||
|       // XXX: event.oldEditor will must probably return a disposed editor. So, it should be treated with care |  | ||||||
|       if (event.newEditor == null) { |       if (event.newEditor == null) { | ||||||
|         (event.oldEditor as? TextEditor)?.editor?.let { |         (event.oldEditor as? TextEditor)?.editor?.let { | ||||||
|           (VimPlugin.getOptionGroup() as OptionGroup).updateFallbackWindow(injector.fallbackWindow, it.vim) |           (VimPlugin.getOptionGroup() as OptionGroup).updateFallbackWindow(injector.fallbackWindow, it.vim) | ||||||
| @@ -66,7 +68,7 @@ internal class OptionGroup : VimOptionGroupBase(), IjVimOptionGroup { | |||||||
| } | } | ||||||
|  |  | ||||||
| internal class IjOptionConstants { | internal class IjOptionConstants { | ||||||
|   @Suppress("SpellCheckingInspection", "MemberVisibilityCanBePrivate", "ConstPropertyName") |   @Suppress("SpellCheckingInspection", "MemberVisibilityCanBePrivate") | ||||||
|   companion object { |   companion object { | ||||||
|  |  | ||||||
|     const val idearefactormode_keep = "keep" |     const val idearefactormode_keep = "keep" | ||||||
|   | |||||||
							
								
								
									
										293
									
								
								src/main/java/com/maddyhome/idea/vim/group/ProcessGroup.java
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										293
									
								
								src/main/java/com/maddyhome/idea/vim/group/ProcessGroup.java
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,293 @@ | |||||||
|  | /* | ||||||
|  |  * Copyright 2003-2023 The IdeaVim authors | ||||||
|  |  * | ||||||
|  |  * Use of this source code is governed by an MIT-style | ||||||
|  |  * license that can be found in the LICENSE.txt file or at | ||||||
|  |  * https://opensource.org/licenses/MIT. | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | package com.maddyhome.idea.vim.group; | ||||||
|  |  | ||||||
|  | import com.intellij.execution.ExecutionException; | ||||||
|  | import com.intellij.execution.configurations.GeneralCommandLine; | ||||||
|  | import com.intellij.execution.process.CapturingProcessHandler; | ||||||
|  | import com.intellij.execution.process.ProcessAdapter; | ||||||
|  | import com.intellij.execution.process.ProcessEvent; | ||||||
|  | import com.intellij.execution.process.ProcessOutput; | ||||||
|  | import com.intellij.openapi.actionSystem.DataContext; | ||||||
|  | import com.intellij.openapi.diagnostic.Logger; | ||||||
|  | import com.intellij.openapi.editor.Editor; | ||||||
|  | import com.intellij.openapi.progress.ProcessCanceledException; | ||||||
|  | import com.intellij.openapi.progress.ProgressIndicator; | ||||||
|  | import com.intellij.openapi.progress.ProgressIndicatorProvider; | ||||||
|  | import com.intellij.openapi.progress.ProgressManager; | ||||||
|  | import com.intellij.util.execution.ParametersListUtil; | ||||||
|  | import com.intellij.util.text.CharSequenceReader; | ||||||
|  | import com.maddyhome.idea.vim.KeyHandler; | ||||||
|  | import com.maddyhome.idea.vim.VimPlugin; | ||||||
|  | import com.maddyhome.idea.vim.api.ExecutionContext; | ||||||
|  | import com.maddyhome.idea.vim.api.VimEditor; | ||||||
|  | import com.maddyhome.idea.vim.api.VimInjectorKt; | ||||||
|  | import com.maddyhome.idea.vim.api.VimProcessGroupBase; | ||||||
|  | import com.maddyhome.idea.vim.command.Command; | ||||||
|  | import com.maddyhome.idea.vim.state.mode.Mode; | ||||||
|  | import com.maddyhome.idea.vim.state.VimStateMachine; | ||||||
|  | import com.maddyhome.idea.vim.ex.ExException; | ||||||
|  | import com.maddyhome.idea.vim.ex.InvalidCommandException; | ||||||
|  | import com.maddyhome.idea.vim.helper.UiHelper; | ||||||
|  | import com.maddyhome.idea.vim.newapi.IjEditorExecutionContext; | ||||||
|  | import com.maddyhome.idea.vim.newapi.IjVimEditor; | ||||||
|  | import com.maddyhome.idea.vim.ui.ex.ExEntryPanel; | ||||||
|  | import com.maddyhome.idea.vim.vimscript.model.CommandLineVimLContext; | ||||||
|  | import org.jetbrains.annotations.NotNull; | ||||||
|  | import org.jetbrains.annotations.Nullable; | ||||||
|  |  | ||||||
|  | import javax.swing.*; | ||||||
|  | import java.io.*; | ||||||
|  | import java.util.ArrayList; | ||||||
|  |  | ||||||
|  | import static com.maddyhome.idea.vim.api.VimInjectorKt.globalOptions; | ||||||
|  | import static com.maddyhome.idea.vim.api.VimInjectorKt.injector; | ||||||
|  |  | ||||||
|  |  | ||||||
|  | public class ProcessGroup extends VimProcessGroupBase { | ||||||
|  |   public String getLastCommand() { | ||||||
|  |     return lastCommand; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @Override | ||||||
|  |   public void startSearchCommand(@NotNull VimEditor editor, ExecutionContext context, int count, char leader) { | ||||||
|  |     if (((IjVimEditor)editor).getEditor().isOneLineMode()) // Don't allow searching in one line editors | ||||||
|  |     { | ||||||
|  |       return; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     String initText = ""; | ||||||
|  |     String label = String.valueOf(leader); | ||||||
|  |  | ||||||
|  |     ExEntryPanel panel = ExEntryPanel.getInstance(); | ||||||
|  |     panel.activate(((IjVimEditor)editor).getEditor(), ((DataContext)context.getContext()), label, initText, count); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @Override | ||||||
|  |   public @NotNull String endSearchCommand() { | ||||||
|  |     ExEntryPanel panel = ExEntryPanel.getInstance(); | ||||||
|  |     panel.deactivate(true); | ||||||
|  |  | ||||||
|  |     return panel.getText(); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   public void startExCommand(@NotNull VimEditor editor, ExecutionContext context, @NotNull Command cmd) { | ||||||
|  |     // Don't allow ex commands in one line editors | ||||||
|  |     if (editor.isOneLineMode()) return; | ||||||
|  |  | ||||||
|  |     String initText = getRange(((IjVimEditor) editor).getEditor(), cmd); | ||||||
|  |     injector.getMarkService().setVisualSelectionMarks(editor); | ||||||
|  |     VimStateMachine.Companion.getInstance(editor).setMode(Mode.CMD_LINE.INSTANCE); | ||||||
|  |     ExEntryPanel panel = ExEntryPanel.getInstance(); | ||||||
|  |     panel.activate(((IjVimEditor) editor).getEditor(), ((IjEditorExecutionContext) context).getContext(), ":", initText, cmd.getCount()); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @Override | ||||||
|  |   public boolean processExKey(@NotNull VimEditor editor, @NotNull KeyStroke stroke) { | ||||||
|  |     // This will only get called if somehow the key focus ended up in the editor while the ex entry window | ||||||
|  |     // is open. So I'll put focus back in the editor and process the key. | ||||||
|  |  | ||||||
|  |     ExEntryPanel panel = ExEntryPanel.getInstance(); | ||||||
|  |     if (panel.isActive()) { | ||||||
|  |       UiHelper.requestFocus(panel.getEntry()); | ||||||
|  |       panel.handleKey(stroke); | ||||||
|  |  | ||||||
|  |       return true; | ||||||
|  |     } | ||||||
|  |     else { | ||||||
|  |       VimStateMachine.Companion.getInstance(editor).setMode(new Mode.NORMAL()); | ||||||
|  |       KeyHandler.getInstance().reset(editor); | ||||||
|  |       return false; | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   public boolean processExEntry(final @NotNull VimEditor editor, final @NotNull ExecutionContext context) { | ||||||
|  |     ExEntryPanel panel = ExEntryPanel.getInstance(); | ||||||
|  |     panel.deactivate(true); | ||||||
|  |     boolean res = true; | ||||||
|  |     try { | ||||||
|  |       VimStateMachine.Companion.getInstance(editor).setMode(new Mode.NORMAL()); | ||||||
|  |  | ||||||
|  |       logger.debug("processing command"); | ||||||
|  |  | ||||||
|  |       String text = panel.getText(); | ||||||
|  |  | ||||||
|  |       if (!panel.getLabel().equals(":")) { | ||||||
|  |         // Search is handled via Argument.Type.EX_STRING. Although ProcessExEntryAction is registered as the handler for | ||||||
|  |         // <CR> in both command and search modes, it's only invoked for command mode (see KeyHandler.handleCommandNode). | ||||||
|  |         // We should never be invoked for anything other than an actual ex command. | ||||||
|  |         throw new InvalidCommandException("Expected ':' command. Got '" + panel.getLabel() + "'", text); | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       if (logger.isDebugEnabled()) logger.debug("swing=" + SwingUtilities.isEventDispatchThread()); | ||||||
|  |  | ||||||
|  |       int repeat = 1; | ||||||
|  |       if (text.contains("raction ")) { | ||||||
|  |         text = text.replace("raction ", "action "); | ||||||
|  |         repeat = panel.getCount(); | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       for (int i = 0; i < repeat; i++) { | ||||||
|  |         VimInjectorKt.getInjector().getVimscriptExecutor().execute(text, editor, context, skipHistory(editor), true, CommandLineVimLContext.INSTANCE); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |     catch (ExException e) { | ||||||
|  |       VimPlugin.showMessage(e.getMessage()); | ||||||
|  |       VimPlugin.indicateError(); | ||||||
|  |       res = false; | ||||||
|  |     } | ||||||
|  |     catch (Exception bad) { | ||||||
|  |       ProcessGroup.logger.error(bad); | ||||||
|  |       VimPlugin.indicateError(); | ||||||
|  |       res = false; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     return res; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   // commands executed from map command / macro should not be added to history | ||||||
|  |   private boolean skipHistory(VimEditor editor) { | ||||||
|  |     return VimStateMachine.Companion.getInstance(editor).getMappingState().isExecutingMap() || injector.getMacro().isExecutingMacro(); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   public void cancelExEntry(final @NotNull VimEditor editor, boolean resetCaret) { | ||||||
|  |     VimStateMachine.Companion.getInstance(editor).setMode(new Mode.NORMAL()); | ||||||
|  |     KeyHandler.getInstance().reset(editor); | ||||||
|  |     ExEntryPanel panel = ExEntryPanel.getInstance(); | ||||||
|  |     panel.deactivate(true, resetCaret); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   @Override | ||||||
|  |   public void startFilterCommand(@NotNull VimEditor editor, ExecutionContext context, @NotNull Command cmd) { | ||||||
|  |     String initText = getRange(((IjVimEditor) editor).getEditor(), cmd) + "!"; | ||||||
|  |     VimStateMachine.Companion.getInstance(editor).setMode(Mode.CMD_LINE.INSTANCE); | ||||||
|  |     ExEntryPanel panel = ExEntryPanel.getInstance(); | ||||||
|  |     panel.activate(((IjVimEditor) editor).getEditor(), ((IjEditorExecutionContext) context).getContext(), ":", initText, 1); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   private @NotNull String getRange(Editor editor, @NotNull Command cmd) { | ||||||
|  |     String initText = ""; | ||||||
|  |     if (VimStateMachine.Companion.getInstance(new IjVimEditor(editor)).getMode() instanceof Mode.VISUAL) { | ||||||
|  |       initText = "'<,'>"; | ||||||
|  |     } | ||||||
|  |     else if (cmd.getRawCount() > 0) { | ||||||
|  |       if (cmd.getCount() == 1) { | ||||||
|  |         initText = "."; | ||||||
|  |       } | ||||||
|  |       else { | ||||||
|  |         initText = ".,.+" + (cmd.getCount() - 1); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     return initText; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   public @Nullable String executeCommand(@NotNull VimEditor editor, @NotNull String command, @Nullable CharSequence input, @Nullable String currentDirectoryPath) | ||||||
|  |     throws ExecutionException, ProcessCanceledException { | ||||||
|  |  | ||||||
|  |     // This is a much simplified version of how Vim does this. We're using stdin/stdout directly, while Vim will | ||||||
|  |     // redirect to temp files ('shellredir' and 'shelltemp') or use pipes. We don't support 'shellquote', because we're | ||||||
|  |     // not handling redirection, but we do use 'shellxquote' and 'shellxescape', because these have defaults that work | ||||||
|  |     // better with Windows. We also don't bother using ShellExecute for Windows commands beginning with `start`. | ||||||
|  |     // Finally, we're also not bothering with the crazy space and backslash handling of the 'shell' options content. | ||||||
|  |     return ProgressManager.getInstance().runProcessWithProgressSynchronously(() -> { | ||||||
|  |  | ||||||
|  |       final String shell = globalOptions(injector).getShell(); | ||||||
|  |       final String shellcmdflag = globalOptions(injector).getShellcmdflag(); | ||||||
|  |       final String shellxescape = globalOptions(injector).getShellxescape(); | ||||||
|  |       final String shellxquote = globalOptions(injector).getShellxquote(); | ||||||
|  |  | ||||||
|  |       // For Win32. See :help 'shellxescape' | ||||||
|  |       final String escapedCommand = shellxquote.equals("(") | ||||||
|  |                                     ? doEscape(command, shellxescape, "^") | ||||||
|  |                                     : command; | ||||||
|  |       // Required for Win32+cmd.exe, defaults to "(". See :help 'shellxquote' | ||||||
|  |       final String quotedCommand = shellxquote.equals("(") | ||||||
|  |                                    ? "(" + escapedCommand + ")" | ||||||
|  |                                    : (shellxquote.equals("\"(") | ||||||
|  |                                       ? "\"(" + escapedCommand + ")\"" | ||||||
|  |                                       : shellxquote + escapedCommand + shellxquote); | ||||||
|  |  | ||||||
|  |       final ArrayList<String> commands = new ArrayList<>(); | ||||||
|  |       commands.add(shell); | ||||||
|  |       if (!shellcmdflag.isEmpty()) { | ||||||
|  |         // Note that Vim also does a simple whitespace split for multiple parameters | ||||||
|  |         commands.addAll(ParametersListUtil.parse(shellcmdflag)); | ||||||
|  |       } | ||||||
|  |       commands.add(quotedCommand); | ||||||
|  |  | ||||||
|  |       if (logger.isDebugEnabled()) { | ||||||
|  |         logger.debug(String.format("shell=%s shellcmdflag=%s command=%s", shell, shellcmdflag, quotedCommand)); | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       final GeneralCommandLine commandLine = new GeneralCommandLine(commands); | ||||||
|  |       if (currentDirectoryPath != null) { | ||||||
|  |         commandLine.setWorkDirectory(currentDirectoryPath); | ||||||
|  |       } | ||||||
|  |       final CapturingProcessHandler handler = new CapturingProcessHandler(commandLine); | ||||||
|  |       if (input != null) { | ||||||
|  |         handler.addProcessListener(new ProcessAdapter() { | ||||||
|  |           @Override | ||||||
|  |           public void startNotified(@NotNull ProcessEvent event) { | ||||||
|  |             try { | ||||||
|  |               final CharSequenceReader charSequenceReader = new CharSequenceReader(input); | ||||||
|  |               final BufferedWriter outputStreamWriter = new BufferedWriter(new OutputStreamWriter(handler.getProcessInput())); | ||||||
|  |               copy(charSequenceReader, outputStreamWriter); | ||||||
|  |               outputStreamWriter.close(); | ||||||
|  |             } | ||||||
|  |             catch (IOException e) { | ||||||
|  |               logger.error(e); | ||||||
|  |             } | ||||||
|  |           } | ||||||
|  |         }); | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       final ProgressIndicator progressIndicator = ProgressIndicatorProvider.getInstance().getProgressIndicator(); | ||||||
|  |       final ProcessOutput output = handler.runProcessWithProgressIndicator(progressIndicator); | ||||||
|  |  | ||||||
|  |       lastCommand = command; | ||||||
|  |  | ||||||
|  |       if (output.isCancelled()) { | ||||||
|  |         // TODO: Vim will use whatever text has already been written to stdout | ||||||
|  |         // For whatever reason, we're not getting any here, so just throw an exception | ||||||
|  |         throw new ProcessCanceledException(); | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       final Integer exitCode = handler.getExitCode(); | ||||||
|  |       if (exitCode != null && exitCode != 0) { | ||||||
|  |         VimPlugin.showMessage("shell returned " + exitCode); | ||||||
|  |         VimPlugin.indicateError(); | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       // Get stderr; stdout and strip colors, which are not handles properly. | ||||||
|  |       return (output.getStderr() + output.getStdout()).replaceAll("\u001B\\[[;\\d]*m", ""); | ||||||
|  |     }, "IdeaVim - !" + command, true, ((IjVimEditor) editor).getEditor().getProject()); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   private String doEscape(String original, String charsToEscape, String escapeChar) { | ||||||
|  |     String result = original; | ||||||
|  |     for (char c : charsToEscape.toCharArray()) { | ||||||
|  |       result = result.replace("" + c, escapeChar + c); | ||||||
|  |     } | ||||||
|  |     return result; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   // TODO: Java 10 has a transferTo method we could use instead | ||||||
|  |   private void copy(@NotNull Reader from, @NotNull Writer to) throws IOException { | ||||||
|  |     char[] buf = new char[2048]; | ||||||
|  |     int cnt; | ||||||
|  |     while ((cnt = from.read(buf)) != -1) { | ||||||
|  |       to.write(buf, 0, cnt); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   private String lastCommand; | ||||||
|  |  | ||||||
|  |   private static final Logger logger = Logger.getInstance(ProcessGroup.class.getName()); | ||||||
|  | } | ||||||
| @@ -1,281 +0,0 @@ | |||||||
| /* |  | ||||||
|  * Copyright 2003-2023 The IdeaVim authors |  | ||||||
|  * |  | ||||||
|  * Use of this source code is governed by an MIT-style |  | ||||||
|  * license that can be found in the LICENSE.txt file or at |  | ||||||
|  * https://opensource.org/licenses/MIT. |  | ||||||
|  */ |  | ||||||
| package com.maddyhome.idea.vim.group |  | ||||||
|  |  | ||||||
| import com.intellij.execution.ExecutionException |  | ||||||
| import com.intellij.execution.configurations.GeneralCommandLine |  | ||||||
| import com.intellij.execution.process.CapturingProcessHandler |  | ||||||
| import com.intellij.execution.process.ProcessAdapter |  | ||||||
| import com.intellij.execution.process.ProcessEvent |  | ||||||
| import com.intellij.openapi.diagnostic.debug |  | ||||||
| import com.intellij.openapi.diagnostic.logger |  | ||||||
| import com.intellij.openapi.progress.ProcessCanceledException |  | ||||||
| import com.intellij.openapi.progress.ProgressIndicatorProvider |  | ||||||
| import com.intellij.openapi.progress.ProgressManager |  | ||||||
| import com.intellij.util.execution.ParametersListUtil |  | ||||||
| import com.intellij.util.text.CharSequenceReader |  | ||||||
| import com.maddyhome.idea.vim.KeyHandler.Companion.getInstance |  | ||||||
| import com.maddyhome.idea.vim.VimPlugin |  | ||||||
| import com.maddyhome.idea.vim.api.ExecutionContext |  | ||||||
| import com.maddyhome.idea.vim.api.VimEditor |  | ||||||
| import com.maddyhome.idea.vim.api.VimProcessGroupBase |  | ||||||
| import com.maddyhome.idea.vim.api.globalOptions |  | ||||||
| import com.maddyhome.idea.vim.api.injector |  | ||||||
| import com.maddyhome.idea.vim.command.Command |  | ||||||
| import com.maddyhome.idea.vim.ex.ExException |  | ||||||
| import com.maddyhome.idea.vim.ex.InvalidCommandException |  | ||||||
| import com.maddyhome.idea.vim.helper.requestFocus |  | ||||||
| import com.maddyhome.idea.vim.helper.vimStateMachine |  | ||||||
| import com.maddyhome.idea.vim.newapi.ij |  | ||||||
| import com.maddyhome.idea.vim.state.VimStateMachine.Companion.getInstance |  | ||||||
| import com.maddyhome.idea.vim.state.mode.Mode |  | ||||||
| import com.maddyhome.idea.vim.state.mode.Mode.NORMAL |  | ||||||
| import com.maddyhome.idea.vim.state.mode.Mode.VISUAL |  | ||||||
| import com.maddyhome.idea.vim.state.mode.ReturnableFromCmd |  | ||||||
| import com.maddyhome.idea.vim.state.mode.mode |  | ||||||
| import com.maddyhome.idea.vim.ui.ex.ExEntryPanel |  | ||||||
| import com.maddyhome.idea.vim.vimscript.model.CommandLineVimLContext |  | ||||||
| import java.io.BufferedWriter |  | ||||||
| import java.io.IOException |  | ||||||
| import java.io.OutputStreamWriter |  | ||||||
| import java.io.Reader |  | ||||||
| import java.io.Writer |  | ||||||
| import javax.swing.KeyStroke |  | ||||||
| import javax.swing.SwingUtilities |  | ||||||
|  |  | ||||||
| public class ProcessGroup : VimProcessGroupBase() { |  | ||||||
|   override var lastCommand: String? = null |  | ||||||
|     private set |  | ||||||
|  |  | ||||||
|   public override fun startSearchCommand(editor: VimEditor, context: ExecutionContext, count: Int, leader: Char) { |  | ||||||
|     // Don't allow searching in one line editors |  | ||||||
|     if (editor.isOneLineMode()) return |  | ||||||
|  |  | ||||||
|     val initText = "" |  | ||||||
|     val label = leader.toString() |  | ||||||
|  |  | ||||||
|     val panel = ExEntryPanel.getInstance() |  | ||||||
|     panel.activate(editor.ij, context.ij, label, initText, count) |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   public override fun endSearchCommand(): String { |  | ||||||
|     val panel = ExEntryPanel.getInstance() |  | ||||||
|     panel.deactivate(true) |  | ||||||
|  |  | ||||||
|     return panel.text |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   public override fun startExCommand(editor: VimEditor, context: ExecutionContext, cmd: Command) { |  | ||||||
|     // Don't allow ex commands in one line editors |  | ||||||
|     if (editor.isOneLineMode()) return |  | ||||||
|  |  | ||||||
|     val currentMode = editor.vimStateMachine.mode |  | ||||||
|     check(currentMode is ReturnableFromCmd) { |  | ||||||
|       "Cannot enable cmd mode from current mode $currentMode" |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     val initText = getRange(editor, cmd) |  | ||||||
|     injector.markService.setVisualSelectionMarks(editor) |  | ||||||
|     editor.vimStateMachine.mode = Mode.CMD_LINE(currentMode) |  | ||||||
|     val panel = ExEntryPanel.getInstance() |  | ||||||
|     panel.activate(editor.ij, context.ij, ":", initText, 1) |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   public override fun processExKey(editor: VimEditor, stroke: KeyStroke): Boolean { |  | ||||||
|     // This will only get called if somehow the key focus ended up in the editor while the ex entry window |  | ||||||
|     // is open. So I'll put focus back in the editor and process the key. |  | ||||||
|  |  | ||||||
|     val panel = ExEntryPanel.getInstance() |  | ||||||
|     if (panel.isActive) { |  | ||||||
|       requestFocus(panel.entry) |  | ||||||
|       panel.handleKey(stroke) |  | ||||||
|  |  | ||||||
|       return true |  | ||||||
|     } else { |  | ||||||
|       getInstance(editor).mode = NORMAL() |  | ||||||
|       getInstance().reset(editor) |  | ||||||
|       return false |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   public override fun processExEntry(editor: VimEditor, context: ExecutionContext): Boolean { |  | ||||||
|     val panel = ExEntryPanel.getInstance() |  | ||||||
|     panel.deactivate(true) |  | ||||||
|     var res = true |  | ||||||
|     try { |  | ||||||
|       getInstance(editor).mode = NORMAL() |  | ||||||
|  |  | ||||||
|       logger.debug("processing command") |  | ||||||
|  |  | ||||||
|       val text = panel.text |  | ||||||
|  |  | ||||||
|       if (panel.label != ":") { |  | ||||||
|         // Search is handled via Argument.Type.EX_STRING. Although ProcessExEntryAction is registered as the handler for |  | ||||||
|         // <CR> in both command and search modes, it's only invoked for command mode (see KeyHandler.handleCommandNode). |  | ||||||
|         // We should never be invoked for anything other than an actual ex command. |  | ||||||
|         throw InvalidCommandException("Expected ':' command. Got '" + panel.label + "'", text) |  | ||||||
|       } |  | ||||||
|  |  | ||||||
|       logger.debug { |  | ||||||
|         "swing=" + SwingUtilities.isEventDispatchThread() |  | ||||||
|       } |  | ||||||
|  |  | ||||||
|       injector.vimscriptExecutor.execute(text, editor, context, skipHistory(editor), true, CommandLineVimLContext) |  | ||||||
|     } catch (e: ExException) { |  | ||||||
|       VimPlugin.showMessage(e.message) |  | ||||||
|       VimPlugin.indicateError() |  | ||||||
|       res = false |  | ||||||
|     } catch (bad: Exception) { |  | ||||||
|       logger.error(bad) |  | ||||||
|       VimPlugin.indicateError() |  | ||||||
|       res = false |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     return res |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   // commands executed from map command / macro should not be added to history |  | ||||||
|   private fun skipHistory(editor: VimEditor): Boolean { |  | ||||||
|     return getInstance(editor).mappingState.isExecutingMap() || injector.macro.isExecutingMacro |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   public override fun cancelExEntry(editor: VimEditor, resetCaret: Boolean) { |  | ||||||
|     editor.vimStateMachine.mode = NORMAL() |  | ||||||
|     getInstance().reset(editor) |  | ||||||
|     val panel = ExEntryPanel.getInstance() |  | ||||||
|     panel.deactivate(true, resetCaret) |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   public override fun startFilterCommand(editor: VimEditor, context: ExecutionContext, cmd: Command) { |  | ||||||
|     val initText = getRange(editor, cmd) + "!" |  | ||||||
|     val currentMode = editor.mode |  | ||||||
|     check(currentMode is ReturnableFromCmd) { "Cannot enable cmd mode from $currentMode" } |  | ||||||
|     editor.vimStateMachine.mode = Mode.CMD_LINE(currentMode) |  | ||||||
|     val panel = ExEntryPanel.getInstance() |  | ||||||
|     panel.activate(editor.ij, context.ij, ":", initText, 1) |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   private fun getRange(editor: VimEditor, cmd: Command): String { |  | ||||||
|     var initText = "" |  | ||||||
|     if (editor.vimStateMachine.mode is VISUAL) { |  | ||||||
|       initText = "'<,'>" |  | ||||||
|     } else if (cmd.rawCount > 0) { |  | ||||||
|       initText = if (cmd.count == 1) { |  | ||||||
|         "." |  | ||||||
|       } else { |  | ||||||
|         ".,.+" + (cmd.count - 1) |  | ||||||
|       } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     return initText |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   @Throws(ExecutionException::class, ProcessCanceledException::class) |  | ||||||
|   public override fun executeCommand( |  | ||||||
|     editor: VimEditor, |  | ||||||
|     command: String, |  | ||||||
|     input: CharSequence?, |  | ||||||
|     currentDirectoryPath: String? |  | ||||||
|   ): String? { |  | ||||||
|     // This is a much simplified version of how Vim does this. We're using stdin/stdout directly, while Vim will |  | ||||||
|     // redirect to temp files ('shellredir' and 'shelltemp') or use pipes. We don't support 'shellquote', because we're |  | ||||||
|     // not handling redirection, but we do use 'shellxquote' and 'shellxescape', because these have defaults that work |  | ||||||
|     // better with Windows. We also don't bother using ShellExecute for Windows commands beginning with `start`. |  | ||||||
|     // Finally, we're also not bothering with the crazy space and backslash handling of the 'shell' options content. |  | ||||||
|  |  | ||||||
|     return ProgressManager.getInstance().runProcessWithProgressSynchronously<String, ExecutionException>( |  | ||||||
|       { |  | ||||||
|         val shell = injector.globalOptions().shell |  | ||||||
|         val shellcmdflag = injector.globalOptions().shellcmdflag |  | ||||||
|         val shellxescape = injector.globalOptions().shellxescape |  | ||||||
|         val shellxquote = injector.globalOptions().shellxquote |  | ||||||
|  |  | ||||||
|         // For Win32. See :help 'shellxescape' |  | ||||||
|         val escapedCommand = if (shellxquote == "(") doEscape(command, shellxescape, "^") |  | ||||||
|         else command |  | ||||||
|         // Required for Win32+cmd.exe, defaults to "(". See :help 'shellxquote' |  | ||||||
|         val quotedCommand = if (shellxquote == "(") "($escapedCommand)" |  | ||||||
|         else (if (shellxquote == "\"(") "\"($escapedCommand)\"" |  | ||||||
|         else shellxquote + escapedCommand + shellxquote) |  | ||||||
|  |  | ||||||
|         val commands = ArrayList<String>() |  | ||||||
|         commands.add(shell) |  | ||||||
|         if (shellcmdflag.isNotEmpty()) { |  | ||||||
|           // Note that Vim also does a simple whitespace split for multiple parameters |  | ||||||
|           commands.addAll(ParametersListUtil.parse(shellcmdflag)) |  | ||||||
|         } |  | ||||||
|         commands.add(quotedCommand) |  | ||||||
|  |  | ||||||
|         if (logger.isDebugEnabled) { |  | ||||||
|           logger.debug(String.format("shell=%s shellcmdflag=%s command=%s", shell, shellcmdflag, quotedCommand)) |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         val commandLine = GeneralCommandLine(commands) |  | ||||||
|         if (currentDirectoryPath != null) { |  | ||||||
|           commandLine.setWorkDirectory(currentDirectoryPath) |  | ||||||
|         } |  | ||||||
|         val handler = CapturingProcessHandler(commandLine) |  | ||||||
|         if (input != null) { |  | ||||||
|           handler.addProcessListener(object : ProcessAdapter() { |  | ||||||
|             override fun startNotified(event: ProcessEvent) { |  | ||||||
|               try { |  | ||||||
|                 val charSequenceReader = CharSequenceReader(input) |  | ||||||
|                 val outputStreamWriter = BufferedWriter(OutputStreamWriter(handler.processInput)) |  | ||||||
|                 copy(charSequenceReader, outputStreamWriter) |  | ||||||
|                 outputStreamWriter.close() |  | ||||||
|               } catch (e: IOException) { |  | ||||||
|                 logger.error(e) |  | ||||||
|               } |  | ||||||
|             } |  | ||||||
|           }) |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         val progressIndicator = ProgressIndicatorProvider.getInstance().progressIndicator |  | ||||||
|         val output = handler.runProcessWithProgressIndicator(progressIndicator) |  | ||||||
|  |  | ||||||
|         lastCommand = command |  | ||||||
|  |  | ||||||
|         if (output.isCancelled) { |  | ||||||
|           // TODO: Vim will use whatever text has already been written to stdout |  | ||||||
|           // For whatever reason, we're not getting any here, so just throw an exception |  | ||||||
|           throw ProcessCanceledException() |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         val exitCode = handler.exitCode |  | ||||||
|         if (exitCode != null && exitCode != 0) { |  | ||||||
|           VimPlugin.showMessage("shell returned $exitCode") |  | ||||||
|           VimPlugin.indicateError() |  | ||||||
|         } |  | ||||||
|         (output.stderr + output.stdout).replace("\u001B\\[[;\\d]*m".toRegex(), "") |  | ||||||
|       }, "IdeaVim - !$command", true, editor.ij.project |  | ||||||
|     ) |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   @Suppress("SameParameterValue") |  | ||||||
|   private fun doEscape(original: String, charsToEscape: String, escapeChar: String): String { |  | ||||||
|     var result = original |  | ||||||
|     for (c in charsToEscape.toCharArray()) { |  | ||||||
|       result = result.replace("" + c, escapeChar + c) |  | ||||||
|     } |  | ||||||
|     return result |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   // TODO: Java 10 has a transferTo method we could use instead |  | ||||||
|   @Throws(IOException::class) |  | ||||||
|   private fun copy(from: Reader, to: Writer) { |  | ||||||
|     val buf = CharArray(2048) |  | ||||||
|     var cnt: Int |  | ||||||
|     while ((from.read(buf).also { cnt = it }) != -1) { |  | ||||||
|       to.write(buf, 0, cnt) |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   public companion object { |  | ||||||
|     private val logger = logger<ProcessGroup>() |  | ||||||
|   } |  | ||||||
| } |  | ||||||
| @@ -15,7 +15,6 @@ import com.intellij.openapi.components.Storage | |||||||
| import com.intellij.openapi.fileEditor.ex.IdeDocumentHistory | import com.intellij.openapi.fileEditor.ex.IdeDocumentHistory | ||||||
| import com.intellij.openapi.fileEditor.impl.IdeDocumentHistoryImpl.PlaceInfo | import com.intellij.openapi.fileEditor.impl.IdeDocumentHistoryImpl.PlaceInfo | ||||||
| import com.intellij.openapi.fileEditor.impl.IdeDocumentHistoryImpl.RecentPlacesListener | import com.intellij.openapi.fileEditor.impl.IdeDocumentHistoryImpl.RecentPlacesListener | ||||||
| import com.intellij.openapi.project.Project |  | ||||||
| import com.intellij.openapi.util.text.StringUtil | import com.intellij.openapi.util.text.StringUtil | ||||||
| import com.maddyhome.idea.vim.api.VimEditor | import com.maddyhome.idea.vim.api.VimEditor | ||||||
| import com.maddyhome.idea.vim.api.VimJumpServiceBase | import com.maddyhome.idea.vim.api.VimJumpServiceBase | ||||||
| @@ -41,34 +40,24 @@ internal class VimJumpServiceImpl : VimJumpServiceBase(), PersistentStateCompone | |||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   // We do not delete old project records. |  | ||||||
|   // Rationale: It's more likely that users will want to review their old projects and access their jump history  |  | ||||||
|   // (e.g., recent files), than for the 100 jumps (max number of records) to consume enough space to be noticeable. |  | ||||||
|   override fun getState(): Element { |   override fun getState(): Element { | ||||||
|     val projectsElem = Element("projects") |     val jumpsElem = Element("jumps") | ||||||
|     for ((project, jumps) in projectToJumps) { |  | ||||||
|       val projectElement = Element("project").setAttribute("id", project) |  | ||||||
|     for (jump in jumps) { |     for (jump in jumps) { | ||||||
|       val jumpElem = Element("jump") |       val jumpElem = Element("jump") | ||||||
|       jumpElem.setAttribute("line", jump.line.toString()) |       jumpElem.setAttribute("line", jump.line.toString()) | ||||||
|       jumpElem.setAttribute("column", jump.col.toString()) |       jumpElem.setAttribute("column", jump.col.toString()) | ||||||
|       jumpElem.setAttribute("filename", StringUtil.notNullize(jump.filepath)) |       jumpElem.setAttribute("filename", StringUtil.notNullize(jump.filepath)) | ||||||
|         projectElement.addContent(jumpElem) |       jumpsElem.addContent(jumpElem) | ||||||
|       if (logger.isDebug()) { |       if (logger.isDebug()) { | ||||||
|         logger.debug("saved jump = $jump") |         logger.debug("saved jump = $jump") | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|       projectsElem.addContent(projectElement) |     return jumpsElem | ||||||
|     } |  | ||||||
|     return projectsElem |  | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   override fun loadState(state: Element) { |   override fun loadState(state: Element) { | ||||||
|     val projectElements = state.getChildren("project") |     val jumpList = state.getChildren("jump") | ||||||
|     for (projectElement in projectElements) { |     for (jumpElement in jumpList) { | ||||||
|       val jumps = mutableListOf<Jump>() |  | ||||||
|       val jumpElements = projectElement.getChildren("jump") |  | ||||||
|       for (jumpElement in jumpElements) { |  | ||||||
|       val jump = Jump( |       val jump = Jump( | ||||||
|         Integer.parseInt(jumpElement.getAttributeValue("line")), |         Integer.parseInt(jumpElement.getAttributeValue("line")), | ||||||
|         Integer.parseInt(jumpElement.getAttributeValue("column")), |         Integer.parseInt(jumpElement.getAttributeValue("column")), | ||||||
| @@ -76,16 +65,14 @@ internal class VimJumpServiceImpl : VimJumpServiceBase(), PersistentStateCompone | |||||||
|       ) |       ) | ||||||
|       jumps.add(jump) |       jumps.add(jump) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     if (logger.isDebug()) { |     if (logger.isDebug()) { | ||||||
|       logger.debug("jumps=$jumps") |       logger.debug("jumps=$jumps") | ||||||
|     } |     } | ||||||
|       val projectId = projectElement.getAttributeValue("id") |  | ||||||
|       projectToJumps[projectId] = jumps |  | ||||||
|     } |  | ||||||
|   } |   } | ||||||
| } | } | ||||||
|  |  | ||||||
| internal class JumpsListener(val project: Project) : RecentPlacesListener { | internal class JumpsListener : RecentPlacesListener { | ||||||
|   override fun recentPlaceAdded(changePlace: PlaceInfo, isChanged: Boolean) { |   override fun recentPlaceAdded(changePlace: PlaceInfo, isChanged: Boolean) { | ||||||
|     if (!injector.globalIjOptions().unifyjumps) return |     if (!injector.globalIjOptions().unifyjumps) return | ||||||
|  |  | ||||||
| @@ -94,7 +81,7 @@ internal class JumpsListener(val project: Project) : RecentPlacesListener { | |||||||
|       if (changePlace.timeStamp < jumpService.lastJumpTimeStamp) return // this listener is notified asynchronously, and |       if (changePlace.timeStamp < jumpService.lastJumpTimeStamp) return // this listener is notified asynchronously, and | ||||||
|       // we do not want jumps that were processed before |       // we do not want jumps that were processed before | ||||||
|       val jump = buildJump(changePlace) ?: return |       val jump = buildJump(changePlace) ?: return | ||||||
|       jumpService.addJump(project.basePath ?: IjVimEditor.DEFAULT_PROJECT_ID, jump, true) |       jumpService.addJump(jump, true) | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
| @@ -106,7 +93,7 @@ internal class JumpsListener(val project: Project) : RecentPlacesListener { | |||||||
|       if (changePlace.timeStamp < jumpService.lastJumpTimeStamp) return // this listener is notified asynchronously, and |       if (changePlace.timeStamp < jumpService.lastJumpTimeStamp) return // this listener is notified asynchronously, and | ||||||
|       // we do not want jumps that were processed before |       // we do not want jumps that were processed before | ||||||
|       val jump = buildJump(changePlace) ?: return |       val jump = buildJump(changePlace) ?: return | ||||||
|       jumpService.removeJump(project.basePath ?: IjVimEditor.DEFAULT_PROJECT_ID, jump) |       jumpService.removeJump(jump) | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -205,7 +205,7 @@ internal class VimMarkServiceImpl : VimMarkServiceBase(), PersistentStateCompone | |||||||
|      * @param event The change event |      * @param event The change event | ||||||
|      */ |      */ | ||||||
|     override fun beforeDocumentChange(event: DocumentEvent) { |     override fun beforeDocumentChange(event: DocumentEvent) { | ||||||
|       if (VimPlugin.isNotEnabled()) return |       if (!VimPlugin.isEnabled()) return | ||||||
|       if (logger.isDebugEnabled) logger.debug("MarkUpdater before, event = $event") |       if (logger.isDebugEnabled) logger.debug("MarkUpdater before, event = $event") | ||||||
|       if (event.oldLength == 0) return |       if (event.oldLength == 0) return | ||||||
|       val doc = event.document |       val doc = event.document | ||||||
| @@ -221,7 +221,7 @@ internal class VimMarkServiceImpl : VimMarkServiceBase(), PersistentStateCompone | |||||||
|      * @param event The change event |      * @param event The change event | ||||||
|      */ |      */ | ||||||
|     override fun documentChanged(event: DocumentEvent) { |     override fun documentChanged(event: DocumentEvent) { | ||||||
|       if (VimPlugin.isNotEnabled()) return |       if (!VimPlugin.isEnabled()) return | ||||||
|       if (logger.isDebugEnabled) logger.debug("MarkUpdater after, event = $event") |       if (logger.isDebugEnabled) logger.debug("MarkUpdater after, event = $event") | ||||||
|       if (event.newLength == 0 || event.newLength == 1 && event.newFragment[0] != '\n') return |       if (event.newLength == 0 || event.newLength == 1 && event.newFragment[0] != '\n') return | ||||||
|       val doc = event.document |       val doc = event.document | ||||||
| @@ -242,7 +242,7 @@ internal class VimMarkServiceImpl : VimMarkServiceBase(), PersistentStateCompone | |||||||
|  |  | ||||||
|   class VimBookmarksListener(private val myProject: Project) : BookmarksListener { |   class VimBookmarksListener(private val myProject: Project) : BookmarksListener { | ||||||
|     override fun bookmarkAdded(group: BookmarkGroup, bookmark: Bookmark) { |     override fun bookmarkAdded(group: BookmarkGroup, bookmark: Bookmark) { | ||||||
|       if (VimPlugin.isNotEnabled()) return |       if (!VimPlugin.isEnabled()) return | ||||||
|       if (!injector.globalIjOptions().ideamarks) { |       if (!injector.globalIjOptions().ideamarks) { | ||||||
|         return |         return | ||||||
|       } |       } | ||||||
| @@ -255,7 +255,7 @@ internal class VimMarkServiceImpl : VimMarkServiceBase(), PersistentStateCompone | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     override fun bookmarkRemoved(group: BookmarkGroup, bookmark: Bookmark) { |     override fun bookmarkRemoved(group: BookmarkGroup, bookmark: Bookmark) { | ||||||
|       if (VimPlugin.isNotEnabled()) return |       if (!VimPlugin.isEnabled()) return | ||||||
|       if (!injector.globalIjOptions().ideamarks) { |       if (!injector.globalIjOptions().ideamarks) { | ||||||
|         return |         return | ||||||
|       } |       } | ||||||
|   | |||||||
| @@ -27,12 +27,15 @@ import com.maddyhome.idea.vim.api.getLineEndOffset | |||||||
| import com.maddyhome.idea.vim.api.globalOptions | import com.maddyhome.idea.vim.api.globalOptions | ||||||
| import com.maddyhome.idea.vim.api.injector | import com.maddyhome.idea.vim.api.injector | ||||||
| import com.maddyhome.idea.vim.api.setChangeMarks | import com.maddyhome.idea.vim.api.setChangeMarks | ||||||
|  | import com.maddyhome.idea.vim.state.mode.SelectionType | ||||||
|  | import com.maddyhome.idea.vim.state.mode.isBlock | ||||||
|  | import com.maddyhome.idea.vim.state.mode.isChar | ||||||
|  | import com.maddyhome.idea.vim.state.mode.isLine | ||||||
| import com.maddyhome.idea.vim.common.TextRange | import com.maddyhome.idea.vim.common.TextRange | ||||||
| import com.maddyhome.idea.vim.diagnostic.debug | import com.maddyhome.idea.vim.diagnostic.debug | ||||||
| import com.maddyhome.idea.vim.helper.EditorHelper | import com.maddyhome.idea.vim.helper.EditorHelper | ||||||
| import com.maddyhome.idea.vim.helper.RWLockLabel | import com.maddyhome.idea.vim.helper.RWLockLabel | ||||||
| import com.maddyhome.idea.vim.helper.moveToInlayAwareOffset | import com.maddyhome.idea.vim.helper.moveToInlayAwareOffset | ||||||
| import com.maddyhome.idea.vim.ide.isClionNova |  | ||||||
| import com.maddyhome.idea.vim.mark.VimMarkConstants.MARK_CHANGE_POS | import com.maddyhome.idea.vim.mark.VimMarkConstants.MARK_CHANGE_POS | ||||||
| import com.maddyhome.idea.vim.newapi.IjVimCaret | import com.maddyhome.idea.vim.newapi.IjVimCaret | ||||||
| import com.maddyhome.idea.vim.newapi.IjVimEditor | import com.maddyhome.idea.vim.newapi.IjVimEditor | ||||||
| @@ -45,10 +48,6 @@ import com.maddyhome.idea.vim.put.PutData | |||||||
| import com.maddyhome.idea.vim.put.VimPasteProvider | import com.maddyhome.idea.vim.put.VimPasteProvider | ||||||
| import com.maddyhome.idea.vim.put.VimPutBase | import com.maddyhome.idea.vim.put.VimPutBase | ||||||
| import com.maddyhome.idea.vim.register.RegisterConstants | import com.maddyhome.idea.vim.register.RegisterConstants | ||||||
| import com.maddyhome.idea.vim.state.mode.SelectionType |  | ||||||
| import com.maddyhome.idea.vim.state.mode.isBlock |  | ||||||
| import com.maddyhome.idea.vim.state.mode.isChar |  | ||||||
| import com.maddyhome.idea.vim.state.mode.isLine |  | ||||||
| import java.awt.datatransfer.DataFlavor | import java.awt.datatransfer.DataFlavor | ||||||
|  |  | ||||||
| internal class PutGroup : VimPutBase() { | internal class PutGroup : VimPutBase() { | ||||||
| @@ -190,7 +189,7 @@ internal class PutGroup : VimPutBase() { | |||||||
|     endOffset: Int, |     endOffset: Int, | ||||||
|   ): Int { |   ): Int { | ||||||
|     // Temp fix for VIM-2808. Should be removed after rider will fix it's issues |     // Temp fix for VIM-2808. Should be removed after rider will fix it's issues | ||||||
|     if (PlatformUtils.isRider() || isClionNova()) return endOffset |     if (PlatformUtils.isRider()) return endOffset | ||||||
|  |  | ||||||
|     val startLine = editor.offsetToBufferPosition(startOffset).line |     val startLine = editor.offsetToBufferPosition(startOffset).line | ||||||
|     val endLine = editor.offsetToBufferPosition(endOffset - 1).line |     val endLine = editor.offsetToBufferPosition(endOffset - 1).line | ||||||
|   | |||||||
| @@ -40,15 +40,9 @@ internal object IdeaSelectionControl { | |||||||
|    * This method should be in sync with [predictMode] |    * This method should be in sync with [predictMode] | ||||||
|    * |    * | ||||||
|    * Control unexpected (non vim) selection change and adjust a mode to it. The new mode is not enabled immediately, |    * Control unexpected (non vim) selection change and adjust a mode to it. The new mode is not enabled immediately, | ||||||
|    *   but with some delay (using [VimVisualTimer]). The delay is used because some platform functionality |    *   but with some delay (using [VimVisualTimer]) | ||||||
|    *   makes features by using selection. E.g. PyCharm unindent firstly select the indenting then applies delete action. |  | ||||||
|    *   Such "quick" selection breaks IdeaVim behaviour. |  | ||||||
|    * |    * | ||||||
|    * See [VimVisualTimer] to more info. |    * See [VimVisualTimer] to more info. | ||||||
|    * |  | ||||||
|    * XXX: This method can be split into "change calculation" and "change apply". In this way, we would be able |  | ||||||
|    *   to calculate if we need to make a change or not and reduce the number of these calls. |  | ||||||
|    *   If this refactoring ever is applied, please add `assertNull(VimVisualTimer.timer)` to `tearDown` of VimTestCase. |  | ||||||
|    */ |    */ | ||||||
|   fun controlNonVimSelectionChange( |   fun controlNonVimSelectionChange( | ||||||
|     editor: Editor, |     editor: Editor, | ||||||
| @@ -56,7 +50,6 @@ internal object IdeaSelectionControl { | |||||||
|   ) { |   ) { | ||||||
|     VimVisualTimer.singleTask(editor.vim.mode) { initialMode -> |     VimVisualTimer.singleTask(editor.vim.mode) { initialMode -> | ||||||
|  |  | ||||||
|       if (VimPlugin.isNotEnabled()) return@singleTask |  | ||||||
|       if (editor.isIdeaVimDisabledHere) return@singleTask |       if (editor.isIdeaVimDisabledHere) return@singleTask | ||||||
|  |  | ||||||
|       logger.debug("Adjust non-vim selection. Source: $selectionSource, initialMode: $initialMode") |       logger.debug("Adjust non-vim selection. Source: $selectionSource, initialMode: $initialMode") | ||||||
| @@ -128,9 +121,8 @@ internal object IdeaSelectionControl { | |||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   private fun dontChangeMode(editor: Editor): Boolean { |   private fun dontChangeMode(editor: Editor): Boolean = | ||||||
|     return editor.isTemplateActive() && (editor.vim.isIdeaRefactorModeKeep || editor.vim.mode.hasVisualSelection) |     editor.isTemplateActive() && (editor.vim.isIdeaRefactorModeKeep || editor.vim.mode.hasVisualSelection) | ||||||
|   } |  | ||||||
|  |  | ||||||
|   private fun chooseNonSelectionMode(editor: Editor): Mode { |   private fun chooseNonSelectionMode(editor: Editor): Mode { | ||||||
|     val templateActive = editor.isTemplateActive() |     val templateActive = editor.isTemplateActive() | ||||||
|   | |||||||
| @@ -9,10 +9,10 @@ | |||||||
| package com.maddyhome.idea.vim.group.visual | package com.maddyhome.idea.vim.group.visual | ||||||
|  |  | ||||||
| import com.maddyhome.idea.vim.api.injector | import com.maddyhome.idea.vim.api.injector | ||||||
|  | import com.maddyhome.idea.vim.state.mode.Mode | ||||||
| import com.maddyhome.idea.vim.group.visual.VimVisualTimer.mode | import com.maddyhome.idea.vim.group.visual.VimVisualTimer.mode | ||||||
| import com.maddyhome.idea.vim.group.visual.VimVisualTimer.singleTask | import com.maddyhome.idea.vim.group.visual.VimVisualTimer.singleTask | ||||||
| import com.maddyhome.idea.vim.newapi.globalIjOptions | import com.maddyhome.idea.vim.newapi.globalIjOptions | ||||||
| import com.maddyhome.idea.vim.state.mode.Mode |  | ||||||
| import java.awt.event.ActionEvent | import java.awt.event.ActionEvent | ||||||
| import javax.swing.Timer | import javax.swing.Timer | ||||||
|  |  | ||||||
| @@ -79,11 +79,6 @@ internal object VimVisualTimer { | |||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   fun drop() { |  | ||||||
|     swingTimer?.stop() |  | ||||||
|     swingTimer = null |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   inline fun timerAction(task: (initialMode: Mode?) -> Unit) { |   inline fun timerAction(task: (initialMode: Mode?) -> Unit) { | ||||||
|     task(mode) |     task(mode) | ||||||
|     swingTimer = null |     swingTimer = null | ||||||
|   | |||||||
| @@ -12,7 +12,6 @@ import com.intellij.serviceContainer.BaseKeyedLazyInstance | |||||||
| import com.intellij.util.SmartList | import com.intellij.util.SmartList | ||||||
| import com.intellij.util.xmlb.annotations.Attribute | import com.intellij.util.xmlb.annotations.Attribute | ||||||
| import com.maddyhome.idea.vim.command.MappingMode | import com.maddyhome.idea.vim.command.MappingMode | ||||||
| import org.jetbrains.annotations.ApiStatus.ScheduledForRemoval |  | ||||||
| import javax.swing.KeyStroke | import javax.swing.KeyStroke | ||||||
|  |  | ||||||
| /** | /** | ||||||
| @@ -37,8 +36,6 @@ import javax.swing.KeyStroke | |||||||
|  *   The reason is startup performance. Using the extension points you don't even have to load classes of actions. |  *   The reason is startup performance. Using the extension points you don't even have to load classes of actions. | ||||||
|  *   So, all actions are loaded on demand, including classes in classloader. |  *   So, all actions are loaded on demand, including classes in classloader. | ||||||
|  */ |  */ | ||||||
| @Deprecated(message = "Please use CommandOrMotion annotation") |  | ||||||
| @ScheduledForRemoval(inVersion = "2.9.0") |  | ||||||
| internal class ActionBeanClass : BaseKeyedLazyInstance<EditorActionHandlerBase>() { | internal class ActionBeanClass : BaseKeyedLazyInstance<EditorActionHandlerBase>() { | ||||||
|   @Attribute("implementation") |   @Attribute("implementation") | ||||||
|   var implementation: String? = null |   var implementation: String? = null | ||||||
|   | |||||||
| @@ -1,91 +0,0 @@ | |||||||
| /* |  | ||||||
|  * Copyright 2003-2023 The IdeaVim authors |  | ||||||
|  * |  | ||||||
|  * Use of this source code is governed by an MIT-style |  | ||||||
|  * license that can be found in the LICENSE.txt file or at |  | ||||||
|  * https://opensource.org/licenses/MIT. |  | ||||||
|  */ |  | ||||||
|  |  | ||||||
| package com.maddyhome.idea.vim.handler |  | ||||||
|  |  | ||||||
| import com.intellij.openapi.actionSystem.KeyboardShortcut |  | ||||||
| import com.intellij.openapi.diagnostic.logger |  | ||||||
| import com.intellij.openapi.keymap.Keymap |  | ||||||
| import com.intellij.openapi.keymap.KeymapManagerListener |  | ||||||
| import com.intellij.openapi.keymap.ex.KeymapManagerEx |  | ||||||
| import com.intellij.openapi.project.Project |  | ||||||
| import com.intellij.openapi.startup.StartupActivity |  | ||||||
| import com.intellij.util.SingleAlarm |  | ||||||
| import com.jetbrains.rd.util.ConcurrentHashMap |  | ||||||
| import com.maddyhome.idea.vim.VimPlugin |  | ||||||
| import com.maddyhome.idea.vim.api.injector |  | ||||||
| import com.maddyhome.idea.vim.api.key |  | ||||||
|  |  | ||||||
|  |  | ||||||
| // We use alarm with delay to avoid many actions in case many events are fired at the same time |  | ||||||
| // [VERSION UPDATE] 2023.3+ Replace SingleAlarm with coroutine flows https://youtrack.jetbrains.com/articles/IJPL-A-8/Alarm-Alternative |  | ||||||
| internal val correctorRequester = SingleAlarm({ correctCopilotKeymap() }, 1_000) |  | ||||||
|  |  | ||||||
| private val LOG = logger<CopilotKeymapCorrector>() |  | ||||||
|  |  | ||||||
| internal class CopilotKeymapCorrector : StartupActivity { |  | ||||||
|   override fun runActivity(project: Project) { |  | ||||||
|     correctorRequester.request() |  | ||||||
|   } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| internal class IdeaVimCorrectorKeymapChangedListener : KeymapManagerListener { |  | ||||||
|   override fun activeKeymapChanged(keymap: Keymap?) { |  | ||||||
|     correctorRequester.request() |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   override fun shortcutChanged(keymap: Keymap, actionId: String) { |  | ||||||
|     correctorRequester.request() |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   override fun shortcutChanged(keymap: Keymap, actionId: String, fromSettings: Boolean) { |  | ||||||
|     correctorRequester.request() |  | ||||||
|   } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| private val copilotHideActionMap = ConcurrentHashMap<String, Unit>() |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * See VIM-3206 |  | ||||||
|  * The user expected to both copilot suggestion and the insert mode to be exited on a single esc. |  | ||||||
|  * However, for the moment, the first esc hides copilot suggestion and the second one exits insert mode. |  | ||||||
|  * To fix this, we remove the esc shortcut from the copilot action if the IdeaVim is active. |  | ||||||
|  * |  | ||||||
|  * This workaround is not the best solution, however, I don't see the better way with the current architecture of |  | ||||||
|  *   actions and EditorHandlers. Firstly, I wanted to suggest to copilot to migrate to EditorActionHandler as well, |  | ||||||
|  *   but this doesn't seem correct for me because in this case the user will lose an ability to change the shorcut for |  | ||||||
|  *   it. It seems like copilot has a similar problem as we do - we don't want to make a handler for "Editor enter action", |  | ||||||
|  *   but a handler for the esc key press. And, moreover, be able to communicate with other plugins about the ordering. |  | ||||||
|  *   Before this feature is implemented, hiding the copilot suggestion on esc looks like a good workaround. |  | ||||||
|  */ |  | ||||||
| private fun correctCopilotKeymap() { |  | ||||||
|   // This is needed to initialize the injector in case this verification is called to fast |  | ||||||
|   VimPlugin.getInstance() |  | ||||||
|  |  | ||||||
|   if (injector.enabler.isEnabled()) { |  | ||||||
|     val keymap = KeymapManagerEx.getInstanceEx().activeKeymap |  | ||||||
|     val res = keymap.getShortcuts("copilot.disposeInlays") |  | ||||||
|     if (res.isEmpty()) return |  | ||||||
|  |  | ||||||
|  |  | ||||||
|     val escapeShortcut = res.find { it.toString() == "[pressed ESCAPE]" } ?: return |  | ||||||
|     keymap.removeShortcut("copilot.disposeInlays", escapeShortcut) |  | ||||||
|     copilotHideActionMap[keymap.name] = Unit |  | ||||||
|     LOG.info("Remove copilot escape shortcut from keymap ${keymap.name}") |  | ||||||
|   } |  | ||||||
|   else { |  | ||||||
|     copilotHideActionMap.forEach { (name, _) -> |  | ||||||
|       val keymap = KeymapManagerEx.getInstanceEx().getKeymap(name) ?: return@forEach |  | ||||||
|       val currentShortcuts = keymap.getShortcuts("copilot.disposeInlays") |  | ||||||
|       if ("[pressed ESCAPE]" !in currentShortcuts.map { it.toString() }) { |  | ||||||
|         keymap.addShortcut("copilot.disposeInlays", KeyboardShortcut(key("<esc>"), null)) |  | ||||||
|       } |  | ||||||
|       LOG.info("Restore copilot escape shortcut in keymap ${keymap.name}") |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
| } |  | ||||||
| @@ -1,67 +0,0 @@ | |||||||
| /* |  | ||||||
|  * Copyright 2003-2023 The IdeaVim authors |  | ||||||
|  * |  | ||||||
|  * Use of this source code is governed by an MIT-style |  | ||||||
|  * license that can be found in the LICENSE.txt file or at |  | ||||||
|  * https://opensource.org/licenses/MIT. |  | ||||||
|  */ |  | ||||||
|  |  | ||||||
| package com.maddyhome.idea.vim.handler |  | ||||||
|  |  | ||||||
| import com.intellij.openapi.actionSystem.IdeActions |  | ||||||
| import com.intellij.openapi.diagnostic.logger |  | ||||||
| import com.intellij.openapi.editor.actionSystem.EditorActionHandlerBean |  | ||||||
| import com.intellij.openapi.extensions.ExtensionPointName |  | ||||||
| import com.intellij.openapi.keymap.ex.KeymapManagerEx |  | ||||||
| import com.intellij.openapi.project.Project |  | ||||||
| import com.intellij.openapi.startup.ProjectActivity |  | ||||||
| import com.maddyhome.idea.vim.api.key |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Logs the chain of handlers for esc and enter |  | ||||||
|  * |  | ||||||
|  * As we made a migration to the new way of handling esc keys (VIM-2974), we may face several issues around that |  | ||||||
|  * One of the possible issues is that some plugin may also register a shortcut for this key and do not pass |  | ||||||
|  * the control to the next handler. In this way, the esc won't work, but there will be no exceptions. |  | ||||||
|  * |  | ||||||
|  * This is a logger that logs the chain of handlers. |  | ||||||
|  * |  | ||||||
|  * Strictly speaking, such access to the extension point is not allowed by the platform. But we can't do this thing |  | ||||||
|  *   otherwise, so let's use it as long as we can. |  | ||||||
|  */ |  | ||||||
| internal class EditorHandlersChainLogger : ProjectActivity { |  | ||||||
|   @Suppress("UnresolvedPluginConfigReference") |  | ||||||
|   private val editorHandlers = ExtensionPointName<EditorActionHandlerBean>("com.intellij.editorActionHandler") |  | ||||||
|  |  | ||||||
|   override suspend fun execute(project: Project) { |  | ||||||
|     val escHandlers = editorHandlers.extensionList |  | ||||||
|       .filter { it.action == "EditorEscape" } |  | ||||||
|       .joinToString("\n") { it.implementationClass } |  | ||||||
|     val enterHandlers = editorHandlers.extensionList |  | ||||||
|       .filter { it.action == "EditorEnter" } |  | ||||||
|       .joinToString("\n") { it.implementationClass } |  | ||||||
|  |  | ||||||
|     LOG.info("Esc handlers chain:\n$escHandlers") |  | ||||||
|     LOG.info("Enter handlers chain:\n$enterHandlers") |  | ||||||
|  |  | ||||||
|     val keymapManager = KeymapManagerEx.getInstanceEx() |  | ||||||
|     val keymap = keymapManager.activeKeymap |  | ||||||
|     val keymapShortcutsForEsc = keymap.getShortcuts(IdeActions.ACTION_EDITOR_ESCAPE).joinToString() |  | ||||||
|     val keymapShortcutsForEnter = keymap.getShortcuts(IdeActions.ACTION_EDITOR_ENTER).joinToString() |  | ||||||
|  |  | ||||||
|     LOG.info("Active keymap (${keymap.name}) shortcuts for esc: $keymapShortcutsForEsc, Shortcuts for enter: $keymapShortcutsForEnter") |  | ||||||
|  |  | ||||||
|     val actionsForEsc = keymap.getActionIds(key("<esc>")).joinToString("\n") |  | ||||||
|     val actionsForEnter = keymap.getActionIds(key("<enter>")).joinToString("\n") |  | ||||||
|  |  | ||||||
|     LOG.info( |  | ||||||
|       "Also keymap (${keymap.name}) has " + |  | ||||||
|         "the following actions assigned to esc:\n$actionsForEsc " + |  | ||||||
|         "\nand following actions assigned to enter:\n$actionsForEnter" |  | ||||||
|     ) |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   companion object { |  | ||||||
|     val LOG = logger<EditorHandlersChainLogger>() |  | ||||||
|   } |  | ||||||
| } |  | ||||||
| @@ -1,155 +0,0 @@ | |||||||
| /* |  | ||||||
|  * Copyright 2003-2023 The IdeaVim authors |  | ||||||
|  * |  | ||||||
|  * Use of this source code is governed by an MIT-style |  | ||||||
|  * license that can be found in the LICENSE.txt file or at |  | ||||||
|  * https://opensource.org/licenses/MIT. |  | ||||||
|  */ |  | ||||||
|  |  | ||||||
| package com.maddyhome.idea.vim.handler |  | ||||||
|  |  | ||||||
| import com.intellij.openapi.actionSystem.IdeActions |  | ||||||
| import com.intellij.openapi.actionSystem.KeyboardShortcut |  | ||||||
| import com.intellij.openapi.actionSystem.Shortcut |  | ||||||
| import com.intellij.openapi.components.Service |  | ||||||
| import com.intellij.openapi.components.service |  | ||||||
| import com.intellij.openapi.keymap.Keymap |  | ||||||
| import com.intellij.openapi.keymap.KeymapManagerListener |  | ||||||
| import com.intellij.openapi.keymap.ex.KeymapManagerEx |  | ||||||
| import com.intellij.openapi.project.Project |  | ||||||
| import com.intellij.openapi.startup.ProjectActivity |  | ||||||
| import com.maddyhome.idea.vim.VimPlugin |  | ||||||
| import com.maddyhome.idea.vim.api.injector |  | ||||||
| import com.maddyhome.idea.vim.api.key |  | ||||||
| import kotlinx.coroutines.CoroutineScope |  | ||||||
| import kotlinx.coroutines.FlowPreview |  | ||||||
| import kotlinx.coroutines.channels.BufferOverflow |  | ||||||
| import kotlinx.coroutines.flow.MutableSharedFlow |  | ||||||
| import kotlinx.coroutines.flow.collectLatest |  | ||||||
| import kotlinx.coroutines.flow.debounce |  | ||||||
| import kotlinx.coroutines.launch |  | ||||||
| import javax.swing.KeyStroke |  | ||||||
|  |  | ||||||
| // We use alarm with delay to avoid many notifications in case many events are fired at the same time |  | ||||||
| internal val keyCheckRequests = MutableSharedFlow<Unit>(replay=1, onBufferOverflow = BufferOverflow.DROP_OLDEST) |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * This checker verifies that the keymap has a correct configuration that is required for IdeaVim plugin |  | ||||||
|  */ |  | ||||||
| internal class KeymapChecker : ProjectActivity { |  | ||||||
|   override suspend fun execute(project: Project) { |  | ||||||
|     project.service<KeymapCheckerService>().start() |  | ||||||
|     keyCheckRequests.emit(Unit) |  | ||||||
|   } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * At the moment of release 2023.3 there is a problem that starting a coroutine like this |  | ||||||
|  *   right in the project activity will block this project activity in tests. |  | ||||||
|  * To avoid that, there is an intermediate service that will allow to avoid this issue. |  | ||||||
|  * |  | ||||||
|  * However, in general we should start this coroutine right in the [KeymapChecker] |  | ||||||
|  */ |  | ||||||
| @OptIn(FlowPreview::class) |  | ||||||
| @Service(Service.Level.PROJECT) |  | ||||||
| internal class KeymapCheckerService(private val cs: CoroutineScope) { |  | ||||||
|   fun start() { |  | ||||||
|     cs.launch { |  | ||||||
|       keyCheckRequests |  | ||||||
|         .debounce(5_000) |  | ||||||
|         .collectLatest { verifyKeymap() } |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| internal class IdeaVimKeymapChangedListener : KeymapManagerListener { |  | ||||||
|   override fun activeKeymapChanged(keymap: Keymap?) { |  | ||||||
|     check(keyCheckRequests.tryEmit(Unit)) |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   override fun shortcutChanged(keymap: Keymap, actionId: String) { |  | ||||||
|     check(keyCheckRequests.tryEmit(Unit)) |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   override fun shortcutChanged(keymap: Keymap, actionId: String, fromSettings: Boolean) { |  | ||||||
|     check(keyCheckRequests.tryEmit(Unit)) |  | ||||||
|   } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * After migration to the editor action handlers, we have to make sure that the keymap has a correct configuration. |  | ||||||
|  * For example, that esc key is assigned to esc editor action |  | ||||||
|  * |  | ||||||
|  * Usually this is not a problem because this is a standard mapping, but the problem may appear in a misconfiguration |  | ||||||
|  *   like it was in VIM-3204 |  | ||||||
|  */ |  | ||||||
| private fun verifyKeymap() { |  | ||||||
|   // This is needed to initialize the injector in case this verification is called to fast |  | ||||||
|   VimPlugin.getInstance() |  | ||||||
|  |  | ||||||
|   if (!injector.enabler.isEnabled()) return |  | ||||||
|  |  | ||||||
|   val keymap = KeymapManagerEx.getInstanceEx().activeKeymap |  | ||||||
|   val keymapShortcutsForEsc = keymap.getShortcuts(IdeActions.ACTION_EDITOR_ESCAPE) |  | ||||||
|   val keymapShortcutsForEnter = keymap.getShortcuts(IdeActions.ACTION_EDITOR_ENTER) |  | ||||||
|  |  | ||||||
|   val issues = ArrayList<KeyMapIssue>() |  | ||||||
|   val correctShortcutMissing = keymapShortcutsForEsc |  | ||||||
|     .filterIsInstance<KeyboardShortcut>() |  | ||||||
|     .none { it.firstKeyStroke.toString() == "pressed ESCAPE" && it.secondKeyStroke == null } |  | ||||||
|  |  | ||||||
|   // We also check if there are any shortcuts starting from esc and with a second key. This should also be removed. |  | ||||||
|   // For example, VIM-3162 has a case when two escapes were assigned to editor escape action |  | ||||||
|   val shortcutsStartingFromEsc = keymapShortcutsForEsc |  | ||||||
|     .filterIsInstance<KeyboardShortcut>() |  | ||||||
|     .filter { it.firstKeyStroke.toString() == "pressed ESCAPE" && it.secondKeyStroke != null } |  | ||||||
|   if (correctShortcutMissing) { |  | ||||||
|     issues += KeyMapIssue.AddShortcut( |  | ||||||
|       "esc", |  | ||||||
|       "editor escape", |  | ||||||
|       IdeActions.ACTION_EDITOR_ESCAPE, |  | ||||||
|       key("<esc>") |  | ||||||
|     ) |  | ||||||
|   } |  | ||||||
|   shortcutsStartingFromEsc.forEach { |  | ||||||
|     issues += KeyMapIssue.RemoveShortcut("editor escape", IdeActions.ACTION_EDITOR_ESCAPE, it) |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|  |  | ||||||
|   val correctEnterShortcutMissing = keymapShortcutsForEnter |  | ||||||
|     .filterIsInstance<KeyboardShortcut>() |  | ||||||
|     .none { it.firstKeyStroke.toString() == "pressed ENTER" && it.secondKeyStroke == null } |  | ||||||
|   val shortcutsStartingFromEnter = keymapShortcutsForEnter |  | ||||||
|     .filterIsInstance<KeyboardShortcut>() |  | ||||||
|     .filter { it.firstKeyStroke.toString() == "pressed ENTER" && it.secondKeyStroke != null } |  | ||||||
|   if (correctEnterShortcutMissing) { |  | ||||||
|     issues += KeyMapIssue.AddShortcut( |  | ||||||
|       "enter", |  | ||||||
|       "editor enter", |  | ||||||
|       IdeActions.ACTION_EDITOR_ENTER, |  | ||||||
|       key("<enter>") |  | ||||||
|     ) |  | ||||||
|   } |  | ||||||
|   shortcutsStartingFromEnter.forEach { |  | ||||||
|     issues += KeyMapIssue.RemoveShortcut("editor enter", IdeActions.ACTION_EDITOR_ENTER, it) |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   if (issues.isNotEmpty()) { |  | ||||||
|     VimPlugin.getNotifications(null).notifyKeymapIssues(issues) |  | ||||||
|   } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| internal sealed interface KeyMapIssue { |  | ||||||
|   data class AddShortcut( |  | ||||||
|     val key: String, |  | ||||||
|     val action: String, |  | ||||||
|     val actionId: String, |  | ||||||
|     val keyStroke: KeyStroke, |  | ||||||
|   ) : KeyMapIssue |  | ||||||
|  |  | ||||||
|   data class RemoveShortcut( |  | ||||||
|     val action: String, |  | ||||||
|     val actionId: String, |  | ||||||
|     val shortcut: Shortcut, |  | ||||||
|   ): KeyMapIssue |  | ||||||
| } |  | ||||||
| @@ -8,67 +8,27 @@ | |||||||
|  |  | ||||||
| package com.maddyhome.idea.vim.handler | package com.maddyhome.idea.vim.handler | ||||||
|  |  | ||||||
| import com.intellij.codeInsight.editorActions.AutoHardWrapHandler |  | ||||||
| import com.intellij.codeInsight.lookup.LookupManager |  | ||||||
| import com.intellij.formatting.LineWrappingUtil |  | ||||||
| import com.intellij.ide.DataManager |  | ||||||
| import com.intellij.openapi.actionSystem.DataContext | import com.intellij.openapi.actionSystem.DataContext | ||||||
| import com.intellij.openapi.application.ApplicationManager |  | ||||||
| import com.intellij.openapi.application.invokeLater |  | ||||||
| import com.intellij.openapi.diagnostic.logger |  | ||||||
| import com.intellij.openapi.editor.Caret | import com.intellij.openapi.editor.Caret | ||||||
| import com.intellij.openapi.editor.Editor | import com.intellij.openapi.editor.Editor | ||||||
| import com.intellij.openapi.editor.actionSystem.EditorActionHandler | import com.intellij.openapi.editor.actionSystem.EditorActionHandler | ||||||
| import com.intellij.openapi.editor.actions.SplitLineAction |  | ||||||
| import com.intellij.openapi.editor.impl.CaretModelImpl |  | ||||||
| import com.intellij.openapi.fileEditor.FileDocumentManager |  | ||||||
| import com.intellij.openapi.util.Key |  | ||||||
| import com.intellij.openapi.util.UserDataHolder |  | ||||||
| import com.intellij.openapi.util.removeUserData |  | ||||||
| import com.maddyhome.idea.vim.KeyHandler | import com.maddyhome.idea.vim.KeyHandler | ||||||
| import com.maddyhome.idea.vim.VimPlugin | import com.maddyhome.idea.vim.VimPlugin | ||||||
| import com.maddyhome.idea.vim.api.injector | import com.maddyhome.idea.vim.api.injector | ||||||
| import com.maddyhome.idea.vim.api.key | import com.maddyhome.idea.vim.api.key | ||||||
| import com.maddyhome.idea.vim.group.IjOptionConstants | import com.maddyhome.idea.vim.command.CommandState | ||||||
| import com.maddyhome.idea.vim.helper.EditorHelper | import com.maddyhome.idea.vim.helper.mode | ||||||
| import com.maddyhome.idea.vim.helper.inNormalMode | import com.maddyhome.idea.vim.helper.vimStateMachine | ||||||
| import com.maddyhome.idea.vim.helper.isPrimaryEditor |  | ||||||
| import com.maddyhome.idea.vim.helper.updateCaretsVisualAttributes |  | ||||||
| import com.maddyhome.idea.vim.newapi.actionStartedFromVim | import com.maddyhome.idea.vim.newapi.actionStartedFromVim | ||||||
| import com.maddyhome.idea.vim.newapi.globalIjOptions | import com.maddyhome.idea.vim.newapi.globalIjOptions | ||||||
| import com.maddyhome.idea.vim.newapi.vim | import com.maddyhome.idea.vim.newapi.vim | ||||||
| import com.maddyhome.idea.vim.state.mode.Mode |  | ||||||
| import com.maddyhome.idea.vim.state.mode.mode |  | ||||||
| import java.awt.event.KeyEvent | import java.awt.event.KeyEvent | ||||||
| import javax.swing.KeyStroke | import javax.swing.KeyStroke | ||||||
|  |  | ||||||
| internal val commandContinuation = Key.create<EditorActionHandler>("commandContinuation") |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Handler that corrects the shape of the caret in python notebooks. |  | ||||||
|  * |  | ||||||
|  * By default, py notebooks show a thin caret after entering the cell. |  | ||||||
|  *   However, we're in normal mode, so this handler fixes it. |  | ||||||
|  */ |  | ||||||
| internal class CaretShapeEnterEditorHandler(private val nextHandler: EditorActionHandler) : EditorActionHandler() { |  | ||||||
|   override fun doExecute(editor: Editor, caret: Caret?, dataContext: DataContext?) { |  | ||||||
|     if (VimPlugin.isEnabled()) { |  | ||||||
|       invokeLater { |  | ||||||
|         editor.updateCaretsVisualAttributes() |  | ||||||
|       } |  | ||||||
|     } |  | ||||||
|     nextHandler.execute(editor, caret, dataContext) |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   override fun isEnabledForCaret(editor: Editor, caret: Caret, dataContext: DataContext?): Boolean { |  | ||||||
|     return nextHandler.isEnabled(editor, caret, dataContext) |  | ||||||
|   } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * This handler doesn't work in tests for ex commands |  * This handler doesn't work in tests for ex commands | ||||||
|  */ |  */ | ||||||
| internal abstract class OctopusHandler(private val nextHandler: EditorActionHandler?) : EditorActionHandler() { | internal abstract class OctopusHandler(private val nextHandler: EditorActionHandler) : EditorActionHandler() { | ||||||
|  |  | ||||||
|   abstract fun executeHandler(editor: Editor, caret: Caret?, dataContext: DataContext?) |   abstract fun executeHandler(editor: Editor, caret: Caret?, dataContext: DataContext?) | ||||||
|   open fun isHandlerEnabled(editor: Editor, dataContext: DataContext?): Boolean { |   open fun isHandlerEnabled(editor: Editor, dataContext: DataContext?): Boolean { | ||||||
| @@ -77,254 +37,64 @@ internal abstract class OctopusHandler(private val nextHandler: EditorActionHand | |||||||
|  |  | ||||||
|   final override fun doExecute(editor: Editor, caret: Caret?, dataContext: DataContext?) { |   final override fun doExecute(editor: Editor, caret: Caret?, dataContext: DataContext?) { | ||||||
|     if (isThisHandlerEnabled(editor, caret, dataContext)) { |     if (isThisHandlerEnabled(editor, caret, dataContext)) { | ||||||
|       val executeInInvokeLater = executeInInvokeLater(editor) |  | ||||||
|       val executionHandler = { |  | ||||||
|         try { |  | ||||||
|           (dataContext as? UserDataHolder)?.putUserData(commandContinuation, nextHandler) |  | ||||||
|       executeHandler(editor, caret, dataContext) |       executeHandler(editor, caret, dataContext) | ||||||
|         } finally { |  | ||||||
|           (dataContext as? UserDataHolder)?.removeUserData(commandContinuation) |  | ||||||
|         } |  | ||||||
|       } |  | ||||||
|  |  | ||||||
|       if (executeInInvokeLater) { |  | ||||||
|         // This `invokeLater` is used to escape the potential `runForEachCaret` function. |  | ||||||
|         // |  | ||||||
|         // The `runForEachCaret` function is disallowed to be called recursively. However, with this new handler, we lose |  | ||||||
|         //   control if we execute the code inside this function or not. See IDEA-300030 for details. |  | ||||||
|         // This means the code in IdeaVim MUST NOT call `runForEachCaret` function. While this is possible for most cases, |  | ||||||
|         //   the user may make a mapping to some intellij action where the `runForEachCaret` is called. This breaks |  | ||||||
|         //   the condition (see VIM-3103 for example). |  | ||||||
|         // Since we can't make sure we don't execute `runForEachCaret`, we have to "escape" out of this function. This is |  | ||||||
|         //   done by scheduling the execution of our code later via the invokeLater function. |  | ||||||
|         // |  | ||||||
|         // We run this job only once for a primary caret. In the handler itself, we'll multiply the execution by the |  | ||||||
|         //   number of carets. If we run this job for each caret, we may end up in the issue like VIM-3186. |  | ||||||
|         //   However, I think that we may do some refactoring to run this job for each caret (if needed). |  | ||||||
|         // |  | ||||||
|         // For the moment, the known case when the caret is null - work in injected editor - VIM-3195 |  | ||||||
|         if (caret == null || caret == editor.caretModel.primaryCaret) { |  | ||||||
|           ApplicationManager.getApplication().invokeLater(executionHandler) |  | ||||||
|         } |  | ||||||
|     } else { |     } else { | ||||||
|         executionHandler() |       nextHandler.execute(editor, caret, dataContext) | ||||||
|       } |  | ||||||
|     } else { |  | ||||||
|       nextHandler?.execute(editor, caret, dataContext) |  | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   private fun executeInInvokeLater(editor: Editor): Boolean { |   @Suppress("RedundantIf") | ||||||
|     // Currently we have a workaround for the PY console VIM-3157 |  | ||||||
|     val fileName = FileDocumentManager.getInstance().getFile(editor.document)?.name |  | ||||||
|     if ( |  | ||||||
|       fileName == "Python Console.py" || // This is the name in 232+ |  | ||||||
|       fileName == "Python Console" // This is the name in 231 |  | ||||||
|     ) return false |  | ||||||
|     return (editor.caretModel as? CaretModelImpl)?.isIteratingOverCarets ?: true |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   private fun isThisHandlerEnabled(editor: Editor, caret: Caret?, dataContext: DataContext?): Boolean { |   private fun isThisHandlerEnabled(editor: Editor, caret: Caret?, dataContext: DataContext?): Boolean { | ||||||
|     if (VimPlugin.isNotEnabled()) return false |     if (!VimPlugin.isEnabled()) return false | ||||||
|     if (!isHandlerEnabled(editor, dataContext)) return false |     if (!isHandlerEnabled(editor, dataContext)) return false | ||||||
|     if (isNotActualKeyPress(dataContext)) return false |     if (dataContext?.actionStartedFromVim == true) return false | ||||||
|  |     if (!enableOctopus) return false | ||||||
|     return true |     return true | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   /** |  | ||||||
|    * In some cases IJ runs handlers to imitate "enter" or other key. In such cases we should not process it on the |  | ||||||
|    *   IdeaVim side because the user may have mappings on enter the we'll get an unexpected behaviour. |  | ||||||
|    * This method should return true if we detect that this handler is called in such case and this is not an |  | ||||||
|    *   actual keypress from the user. |  | ||||||
|    */ |  | ||||||
|   private fun isNotActualKeyPress(dataContext: DataContext?): Boolean { |  | ||||||
|     if (dataContext != null) { |  | ||||||
|       // This flag is set when the enter handlers are executed as a part of moving the comment on the new line |  | ||||||
|       val dataManager = DataManager.getInstance() |  | ||||||
|       if (dataManager.loadFromDataContext(dataContext, AutoHardWrapHandler.AUTO_WRAP_LINE_IN_PROGRESS_KEY) == true) { |  | ||||||
|         return true |  | ||||||
|       } |  | ||||||
|  |  | ||||||
|       // From VIM-3177 |  | ||||||
|       val wrapLongLineDuringFormattingInProgress = dataManager |  | ||||||
|         .loadFromDataContext(dataContext, LineWrappingUtil.WRAP_LONG_LINE_DURING_FORMATTING_IN_PROGRESS_KEY) |  | ||||||
|       if (wrapLongLineDuringFormattingInProgress == true) { |  | ||||||
|         return true |  | ||||||
|       } |  | ||||||
|  |  | ||||||
|       // From VIM-3203 |  | ||||||
|       val splitLineInProgress = dataManager.loadFromDataContext(dataContext, SplitLineAction.SPLIT_LINE_KEY) |  | ||||||
|       if (splitLineInProgress == true) { |  | ||||||
|         return true |  | ||||||
|       } |  | ||||||
|  |  | ||||||
|       if (dataManager.loadFromDataContext(dataContext, StartNewLineDetectorBase.Util.key) == true) { |  | ||||||
|         return true |  | ||||||
|       } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     if (dataContext?.actionStartedFromVim == true) return true |  | ||||||
|  |  | ||||||
|     return false |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   final override fun isEnabledForCaret(editor: Editor, caret: Caret, dataContext: DataContext?): Boolean { |   final override fun isEnabledForCaret(editor: Editor, caret: Caret, dataContext: DataContext?): Boolean { | ||||||
|     return isThisHandlerEnabled(editor, caret, dataContext) |     return isThisHandlerEnabled(editor, caret, dataContext) || nextHandler.isEnabled(editor, caret, dataContext) | ||||||
|       || nextHandler?.isEnabled(editor, caret, dataContext) == true |  | ||||||
|   } |   } | ||||||
| } | } | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * Known conflicts & solutions: |  * Known conflicts & solutions: | ||||||
|  * - Smart step into - set handler after |  * - Smart step into - set handler after | ||||||
|  * - Python notebooks - set handler after |  * - Python notebooks - set handler before - test needed! | ||||||
|  * - Ace jump - set handler after |  * - Ace jump - set handler after | ||||||
|  * - Lookup - doesn't intersect with enter anymore |  * - Lookup - doesn't intersect with enter anymore | ||||||
|  * - App code - set handler after |  * - App code - set handler after | ||||||
|  * - Template - doesn't intersect with enter anymore |  * - Template - doesn't intersect with enter anymore | ||||||
|  * - rd.client.editor.enter - set handler before. Otherwise, rider will add new line on enter even in normal mode |  | ||||||
|  * |  | ||||||
|  * This rule is disabled due to VIM-3124 |  | ||||||
|  * - before terminalEnter - not necessary, but terminalEnter causes "file is read-only" tooltip for readonly files VIM-3122 |  | ||||||
|  * - `first` is set to satisfy sorting condition "before terminalEnter". |  | ||||||
|  * |  | ||||||
|  * |  | ||||||
|  * DO NOT add handlers that force to add "first" ordering. This doesn't work with jupyterCommandModeEnterKeyHandler (see VIM-3124) |  | ||||||
|  */ |  */ | ||||||
| internal class VimEnterHandler(nextHandler: EditorActionHandler?) : VimKeyHandler(nextHandler) { | internal class VimEnterHandler(nextHandler: EditorActionHandler) : VimKeyHandler(nextHandler) { | ||||||
|   override val key: String = "<CR>" |   override val key: String = "<CR>" | ||||||
|  |  | ||||||
|   override fun isHandlerEnabled(editor: Editor, dataContext: DataContext?): Boolean { |  | ||||||
|     if (!super.isHandlerEnabled(editor, dataContext)) return false |  | ||||||
|     // This is important for one-line editors, to turn off enter. |  | ||||||
|     // Some one-line editors rely on the fact that there are no enter actions registered. For example, hash search in git |  | ||||||
|     // See VIM-2974 for example where it was broken |  | ||||||
|     return !editor.isOneLineMode |  | ||||||
|   } |  | ||||||
| } | } | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * Known conflicts & solutions: |  * Known conflicts & solutions: | ||||||
|  * |  * | ||||||
|  * - Smart step into - set handler after |  * - Smart step into - set handler after | ||||||
|  * - Python notebooks - set handler before - yes, we have `<CR>` as "after" and `<esc>` as before. I'm not completely sure |  * - Python notebooks - set handler before - test needed | ||||||
|  *   why this combination is correct, but other versions don't work. |  | ||||||
|  * - Ace jump - set handler after |  * - Ace jump - set handler after | ||||||
|  * - Lookup - It disappears after putting our esc before templateEscape. But I'm not sure why it works like that |  * - Lookup - It disappears after putting our esc before templateEscape. But I'm not sure why it works like that | ||||||
|  * - App code - Need to review |  * - App code - Need to review | ||||||
|  * - Template - Need to review |  * - Template - Need to review | ||||||
|  * - before backend.escape - to handle our handlers before Rider processing. Also, without this rule, we get problems like VIM-3146 |  | ||||||
|  */ |  */ | ||||||
| internal class VimEscHandler(nextHandler: EditorActionHandler) : VimKeyHandler(nextHandler) { | internal class VimEscHandler(nextHandler: EditorActionHandler) : VimKeyHandler(nextHandler) { | ||||||
|   override val key: String = "<Esc>" |   override val key: String = "<Esc>" | ||||||
|  |  | ||||||
|  |   /** | ||||||
|  |    * Also, we need to pass esc to IDE if we're in normal mode and there is nothing to cancel | ||||||
|  |    * (e.g. we still can cancel numbers, or cancel the replace character mode) | ||||||
|  |    */ | ||||||
|   override fun isHandlerEnabled(editor: Editor, dataContext: DataContext?): Boolean { |   override fun isHandlerEnabled(editor: Editor, dataContext: DataContext?): Boolean { | ||||||
|     val ideaVimSupportDialog = |     return editor.mode != CommandState.Mode.COMMAND || | ||||||
|       injector.globalIjOptions().ideavimsupport.contains(IjOptionConstants.ideavimsupport_dialog) |       editor.vimStateMachine?.commandBuilder?.count != 0 || | ||||||
|  |       editor.vimStateMachine?.isReplaceCharacter == true | ||||||
|     return editor.isPrimaryEditor() || |  | ||||||
|       EditorHelper.isFileEditor(editor) && !editor.vim.mode.inNormalMode || |  | ||||||
|       ideaVimSupportDialog && !editor.vim.mode.inNormalMode |  | ||||||
|   } |   } | ||||||
| } | } | ||||||
|  |  | ||||||
| /** | internal abstract class VimKeyHandler(nextHandler: EditorActionHandler) : OctopusHandler(nextHandler) { | ||||||
|  * Rider (and CLion Nova) uses a separate handler for esc to close the completion. IdeaOnlyEscapeHandlerAction is especially |  | ||||||
|  *   designer to get all the esc presses, and if there is a completion close it and do not pass the execution further. |  | ||||||
|  *   This doesn't work the same as in IJ. |  | ||||||
|  * In IdeaVim, we'd like to exit insert mode on closing completion. This is a requirement as the change of this |  | ||||||
|  *   behaviour causes a lot of complaining from users. Since the rider handler gets execution control, we don't |  | ||||||
|  *    receive an event and don't exit the insert mode. |  | ||||||
|  * To fix it, this special handler exists only for rider and stands before the rider's handler. We don't execute the |  | ||||||
|  *   handler from rider because the autocompletion is closed automatically anyway. |  | ||||||
|  */ |  | ||||||
| internal class VimEscForRiderHandler(nextHandler: EditorActionHandler) : VimKeyHandler(nextHandler) { |  | ||||||
|   override val key: String = "<Esc>" |  | ||||||
|  |  | ||||||
|   override fun isHandlerEnabled(editor: Editor, dataContext: DataContext?): Boolean { |  | ||||||
|     return LookupManager.getActiveLookup(editor) != null |  | ||||||
|   } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Empty logger for esc presses |  | ||||||
|  * |  | ||||||
|  * As we made a migration to the new way of handling esc keys (VIM-2974), we may face several issues around that |  | ||||||
|  * One of the possible issues is that some plugin may also register a shortcut for this key and do not pass |  | ||||||
|  * the control to the next handler. In this way, the esc won't work, but there will be no exceptions. |  | ||||||
|  * This handler, that should stand in front of handlers change, just logs the event of pressing the key |  | ||||||
|  * and passes the execution. |  | ||||||
|  */ |  | ||||||
| internal class VimEscLoggerHandler(private val nextHandler: EditorActionHandler) : EditorActionHandler() { |  | ||||||
|   override fun doExecute(editor: Editor, caret: Caret?, dataContext: DataContext?) { |  | ||||||
|     LOG.info("Esc pressed") |  | ||||||
|     nextHandler.execute(editor, caret, dataContext) |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   override fun isEnabledForCaret(editor: Editor, caret: Caret, dataContext: DataContext?): Boolean { |  | ||||||
|     return nextHandler.isEnabled(editor, caret, dataContext) |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   companion object { |  | ||||||
|     val LOG = logger<VimEscLoggerHandler>() |  | ||||||
|   } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Workaround to support "Start New Line" action in normal mode. |  | ||||||
|  * IJ executes enter handler on "Start New Line". This causes an issue that IdeaVim thinks that this is just an enter key. |  | ||||||
|  * This thing should be refactored, but for now we'll use this workaround VIM-3159 |  | ||||||
|  * |  | ||||||
|  * The Same thing happens with "Start New Line Before Current" action. |  | ||||||
|  */ |  | ||||||
| internal class StartNewLineDetector(nextHandler: EditorActionHandler) : StartNewLineDetectorBase(nextHandler) |  | ||||||
| internal class StartNewLineBeforeCurrentDetector(nextHandler: EditorActionHandler) : |  | ||||||
|   StartNewLineDetectorBase(nextHandler) |  | ||||||
|  |  | ||||||
| internal open class StartNewLineDetectorBase(private val nextHandler: EditorActionHandler) : EditorActionHandler() { |  | ||||||
|   override fun doExecute(editor: Editor, caret: Caret?, dataContext: DataContext?) { |  | ||||||
|     DataManager.getInstance().saveInDataContext(dataContext, Util.key, true) |  | ||||||
|     nextHandler.execute(editor, caret, dataContext) |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   override fun isEnabledForCaret(editor: Editor, caret: Caret, dataContext: DataContext?): Boolean { |  | ||||||
|     return nextHandler.isEnabled(editor, caret, dataContext) |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   object Util { |  | ||||||
|     val key = Key.create<Boolean>("vim.is.start.new.line") |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   companion object { |  | ||||||
|     val LOG = logger<VimEscLoggerHandler>() |  | ||||||
|   } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Empty logger for enter presses |  | ||||||
|  * |  | ||||||
|  * As we made a migration to the new way of handling enter keys (VIM-2974), we may face several issues around that |  | ||||||
|  * One of the possible issues is that some plugin may also register a shortcut for this key and do not pass |  | ||||||
|  * the control to the next handler. In this way, the esc won't work, but there will be no exceptions. |  | ||||||
|  * This handler, that should stand in front of handlers change, just logs the event of pressing the key |  | ||||||
|  * and passes the execution. |  | ||||||
|  */ |  | ||||||
| internal class VimEnterLoggerHandler(private val nextHandler: EditorActionHandler) : EditorActionHandler() { |  | ||||||
|   override fun doExecute(editor: Editor, caret: Caret?, dataContext: DataContext?) { |  | ||||||
|     LOG.info("Enter pressed") |  | ||||||
|     nextHandler.execute(editor, caret, dataContext) |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   override fun isEnabledForCaret(editor: Editor, caret: Caret, dataContext: DataContext?): Boolean { |  | ||||||
|     return nextHandler.isEnabled(editor, caret, dataContext) |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   companion object { |  | ||||||
|     val LOG = logger<VimEnterLoggerHandler>() |  | ||||||
|   } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| internal abstract class VimKeyHandler(nextHandler: EditorActionHandler?) : OctopusHandler(nextHandler) { |  | ||||||
|  |  | ||||||
|   abstract val key: String |   abstract val key: String | ||||||
|  |  | ||||||
| @@ -341,12 +111,27 @@ internal abstract class VimKeyHandler(nextHandler: EditorActionHandler?) : Octop | |||||||
| } | } | ||||||
|  |  | ||||||
| internal fun isOctopusEnabled(s: KeyStroke, editor: Editor): Boolean { | internal fun isOctopusEnabled(s: KeyStroke, editor: Editor): Boolean { | ||||||
|   // CMD line has a different processing mechanizm: the processing actions are registered |   if (!enableOctopus) return false | ||||||
|   //   for the input field component. These keys are not dispatched via the octopus handler. |  | ||||||
|   if (editor.vim.mode is Mode.CMD_LINE) return false |  | ||||||
|   when { |   when { | ||||||
|     s.keyCode == KeyEvent.VK_ENTER && s.modifiers == 0 -> return true |     s.keyCode == KeyEvent.VK_ENTER -> return editor.mode in listOf( | ||||||
|     s.keyCode == KeyEvent.VK_ESCAPE && s.modifiers == 0 -> return true |       CommandState.Mode.COMMAND, | ||||||
|  |       CommandState.Mode.INSERT, | ||||||
|  |       CommandState.Mode.VISUAL, | ||||||
|  |     ) | ||||||
|  |     s.keyCode == KeyEvent.VK_ESCAPE -> return editor.mode in listOf( | ||||||
|  |       CommandState.Mode.COMMAND, | ||||||
|  |       CommandState.Mode.INSERT, | ||||||
|  |       CommandState.Mode.VISUAL, | ||||||
|  |     ) | ||||||
|   } |   } | ||||||
|   return false |   return false | ||||||
| } | } | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * Experiment: At the moment, IdeaVim intersects all shortcuts and sends the to [KeyHandler] | ||||||
|  |  * However, this doesn't seem to be a good solution as other handlers are overridden by vim. | ||||||
|  |  * If this option is enabled, vim will connect to IDE via EditorActionHandler extension point | ||||||
|  |  *   what seems to be a way better solution as this is a correct way to override editor actions like enter, right, etc. | ||||||
|  |  */ | ||||||
|  | internal val enableOctopus: Boolean | ||||||
|  |   get() = injector.globalIjOptions().octopushandler | ||||||
|   | |||||||
| @@ -8,13 +8,11 @@ | |||||||
|  |  | ||||||
| package com.maddyhome.idea.vim.helper | package com.maddyhome.idea.vim.helper | ||||||
|  |  | ||||||
| import com.intellij.openapi.diagnostic.thisLogger |  | ||||||
| import com.intellij.openapi.editor.Caret | import com.intellij.openapi.editor.Caret | ||||||
| import com.intellij.openapi.editor.CaretVisualAttributes | import com.intellij.openapi.editor.CaretVisualAttributes | ||||||
| import com.intellij.openapi.editor.Editor | import com.intellij.openapi.editor.Editor | ||||||
| import com.intellij.openapi.editor.ex.EditorEx | import com.intellij.openapi.editor.ex.EditorEx | ||||||
| import com.intellij.openapi.editor.ex.EditorSettingsExternalizable | import com.intellij.openapi.editor.ex.EditorSettingsExternalizable | ||||||
| import com.maddyhome.idea.vim.VimPlugin |  | ||||||
| import com.maddyhome.idea.vim.api.VimEditor | import com.maddyhome.idea.vim.api.VimEditor | ||||||
| import com.maddyhome.idea.vim.api.globalOptions | import com.maddyhome.idea.vim.api.globalOptions | ||||||
| import com.maddyhome.idea.vim.api.injector | import com.maddyhome.idea.vim.api.injector | ||||||
| @@ -81,7 +79,6 @@ private fun Editor.guicursorMode(): GuiCursorMode { | |||||||
| private fun isBlockCursorOverride() = EditorSettingsExternalizable.getInstance().isBlockCursor | private fun isBlockCursorOverride() = EditorSettingsExternalizable.getInstance().isBlockCursor | ||||||
|  |  | ||||||
| private fun Editor.updatePrimaryCaretVisualAttributes() { | private fun Editor.updatePrimaryCaretVisualAttributes() { | ||||||
|   if (VimPlugin.isNotEnabled()) thisLogger().error("The caret attributes should not be updated if the IdeaVim is disabled") |  | ||||||
|   caretModel.primaryCaret.visualAttributes = AttributesCache.getCaretVisualAttributes(this) |   caretModel.primaryCaret.visualAttributes = AttributesCache.getCaretVisualAttributes(this) | ||||||
|  |  | ||||||
|   // Make sure the caret is visible as soon as it's set. It might be invisible while blinking |   // Make sure the caret is visible as soon as it's set. It might be invisible while blinking | ||||||
| @@ -89,7 +86,6 @@ private fun Editor.updatePrimaryCaretVisualAttributes() { | |||||||
| } | } | ||||||
|  |  | ||||||
| private fun Editor.updateSecondaryCaretsVisualAttributes() { | private fun Editor.updateSecondaryCaretsVisualAttributes() { | ||||||
|   if (VimPlugin.isNotEnabled()) thisLogger().error("The caret attributes should not be updated if the IdeaVim is disabled") |  | ||||||
|   // IntelliJ simulates visual block with multiple carets with selections. Do our best to hide them |   // IntelliJ simulates visual block with multiple carets with selections. Do our best to hide them | ||||||
|   val attributes = if (this.vim.inBlockSelection) HIDDEN else AttributesCache.getCaretVisualAttributes(this) |   val attributes = if (this.vim.inBlockSelection) HIDDEN else AttributesCache.getCaretVisualAttributes(this) | ||||||
|   this.caretModel.allCarets.forEach { |   this.caretModel.allCarets.forEach { | ||||||
|   | |||||||
| @@ -36,7 +36,7 @@ public val Editor.mode: CommandState.Mode | |||||||
|   get() { |   get() { | ||||||
|     val mode = this.vim.vimStateMachine.mode |     val mode = this.vim.vimStateMachine.mode | ||||||
|     return when (mode) { |     return when (mode) { | ||||||
|       is Mode.CMD_LINE -> CommandState.Mode.CMD_LINE |       Mode.CMD_LINE -> CommandState.Mode.CMD_LINE | ||||||
|       Mode.INSERT -> CommandState.Mode.INSERT |       Mode.INSERT -> CommandState.Mode.INSERT | ||||||
|       is Mode.NORMAL -> CommandState.Mode.COMMAND |       is Mode.NORMAL -> CommandState.Mode.COMMAND | ||||||
|       is Mode.OP_PENDING -> CommandState.Mode.OP_PENDING |       is Mode.OP_PENDING -> CommandState.Mode.OP_PENDING | ||||||
|   | |||||||
| @@ -335,7 +335,7 @@ public class EditorHelper { | |||||||
|  |  | ||||||
|     final int offset = y - ((screenHeight - lineHeight) / lineHeight / 2 * lineHeight); |     final int offset = y - ((screenHeight - lineHeight) / lineHeight / 2 * lineHeight); | ||||||
|     @NotNull final VimEditor editor1 = new IjVimEditor(editor); |     @NotNull final VimEditor editor1 = new IjVimEditor(editor); | ||||||
|     final int lastVisualLine = EngineEditorHelperKt.getVisualLineCount(editor1) - 1; |     final int lastVisualLine = EngineEditorHelperKt.getVisualLineCount(editor1) + editor.getSettings().getAdditionalLinesCount(); | ||||||
|     final int offsetForLastLineAtBottom = getOffsetToScrollVisualLineToBottomOfScreen(editor, lastVisualLine); |     final int offsetForLastLineAtBottom = getOffsetToScrollVisualLineToBottomOfScreen(editor, lastVisualLine); | ||||||
|  |  | ||||||
|     // For `zz`, we want to use virtual space and move any line, including the last one, to the middle of the screen. |     // For `zz`, we want to use virtual space and move any line, including the last one, to the middle of the screen. | ||||||
|   | |||||||
| @@ -12,6 +12,7 @@ package com.maddyhome.idea.vim.helper | |||||||
|  |  | ||||||
| import com.intellij.codeWithMe.ClientId | import com.intellij.codeWithMe.ClientId | ||||||
| import com.intellij.openapi.editor.Caret | import com.intellij.openapi.editor.Caret | ||||||
|  | import com.intellij.openapi.editor.CaretState | ||||||
| import com.intellij.openapi.editor.Editor | import com.intellij.openapi.editor.Editor | ||||||
| import com.intellij.openapi.editor.ex.util.EditorUtil | import com.intellij.openapi.editor.ex.util.EditorUtil | ||||||
| import com.intellij.openapi.fileEditor.ex.FileEditorManagerEx | import com.intellij.openapi.fileEditor.ex.FileEditorManagerEx | ||||||
| @@ -19,6 +20,8 @@ import com.intellij.util.ui.table.JBTableRowEditor | |||||||
| import com.maddyhome.idea.vim.api.injector | import com.maddyhome.idea.vim.api.injector | ||||||
| import com.maddyhome.idea.vim.group.IjOptionConstants | import com.maddyhome.idea.vim.group.IjOptionConstants | ||||||
| import com.maddyhome.idea.vim.newapi.globalIjOptions | import com.maddyhome.idea.vim.newapi.globalIjOptions | ||||||
|  | import com.maddyhome.idea.vim.newapi.vim | ||||||
|  | import com.maddyhome.idea.vim.state.mode.inBlockSelection | ||||||
| import java.awt.Component | import java.awt.Component | ||||||
| import javax.swing.JComponent | import javax.swing.JComponent | ||||||
| import javax.swing.JTable | import javax.swing.JTable | ||||||
| @@ -93,3 +96,41 @@ internal val Caret.vimLine: Int | |||||||
|  */ |  */ | ||||||
| internal val Editor.vimLine: Int | internal val Editor.vimLine: Int | ||||||
|   get() = this.caretModel.currentCaret.vimLine |   get() = this.caretModel.currentCaret.vimLine | ||||||
|  |  | ||||||
|  | internal inline fun Editor.runWithEveryCaretAndRestore(action: () -> Unit) { | ||||||
|  |   val caretModel = this.caretModel | ||||||
|  |   val carets = if (this.vim.inBlockSelection) null else caretModel.allCarets | ||||||
|  |   if (carets == null || carets.size == 1) { | ||||||
|  |     action() | ||||||
|  |   } | ||||||
|  |   else { | ||||||
|  |     var initialDocumentSize = this.document.textLength | ||||||
|  |     var documentSizeDifference = 0 | ||||||
|  |  | ||||||
|  |     val caretOffsets = carets.map { it.selectionStart to it.selectionEnd } | ||||||
|  |     val restoredCarets = mutableListOf<CaretState>() | ||||||
|  |  | ||||||
|  |     caretModel.removeSecondaryCarets() | ||||||
|  |      | ||||||
|  |     for ((selectionStart, selectionEnd) in caretOffsets) { | ||||||
|  |       if (selectionStart == selectionEnd) { | ||||||
|  |         caretModel.primaryCaret.moveToOffset(selectionStart + documentSizeDifference) | ||||||
|  |       } | ||||||
|  |       else { | ||||||
|  |         caretModel.primaryCaret.setSelection( | ||||||
|  |           selectionStart + documentSizeDifference, | ||||||
|  |           selectionEnd + documentSizeDifference | ||||||
|  |         ) | ||||||
|  |       } | ||||||
|  |        | ||||||
|  |       action() | ||||||
|  |       restoredCarets.add(caretModel.caretsAndSelections.single()) | ||||||
|  |  | ||||||
|  |       val documentLength = this.document.textLength | ||||||
|  |       documentSizeDifference += documentLength - initialDocumentSize | ||||||
|  |       initialDocumentSize = documentLength | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     caretModel.caretsAndSelections = restoredCarets | ||||||
|  |   }  | ||||||
|  | } | ||||||
|   | |||||||
| @@ -110,7 +110,7 @@ internal fun Editor.isTemplateActive(): Boolean { | |||||||
| } | } | ||||||
|  |  | ||||||
| private fun vimEnabled(editor: Editor?): Boolean { | private fun vimEnabled(editor: Editor?): Boolean { | ||||||
|   if (VimPlugin.isNotEnabled()) return false |   if (!VimPlugin.isEnabled()) return false | ||||||
|   if (editor != null && editor.isIdeaVimDisabledHere) return false |   if (editor != null && editor.isIdeaVimDisabledHere) return false | ||||||
|   return true |   return true | ||||||
| } | } | ||||||
|   | |||||||
| @@ -15,12 +15,10 @@ import com.intellij.openapi.actionSystem.AnAction | |||||||
| import com.intellij.openapi.actionSystem.AnActionEvent | import com.intellij.openapi.actionSystem.AnActionEvent | ||||||
| import com.intellij.openapi.actionSystem.AnActionResult | import com.intellij.openapi.actionSystem.AnActionResult | ||||||
| import com.intellij.openapi.actionSystem.DataContextWrapper | import com.intellij.openapi.actionSystem.DataContextWrapper | ||||||
| import com.intellij.openapi.actionSystem.EmptyAction |  | ||||||
| import com.intellij.openapi.actionSystem.IdeActions | import com.intellij.openapi.actionSystem.IdeActions | ||||||
| import com.intellij.openapi.actionSystem.PlatformDataKeys | import com.intellij.openapi.actionSystem.PlatformDataKeys | ||||||
| import com.intellij.openapi.actionSystem.ex.ActionManagerEx | import com.intellij.openapi.actionSystem.ex.ActionManagerEx | ||||||
| import com.intellij.openapi.actionSystem.ex.ActionUtil | import com.intellij.openapi.actionSystem.ex.ActionUtil | ||||||
| import com.intellij.openapi.actionSystem.impl.ProxyShortcutSet |  | ||||||
| import com.intellij.openapi.command.CommandProcessor | import com.intellij.openapi.command.CommandProcessor | ||||||
| import com.intellij.openapi.command.UndoConfirmationPolicy | import com.intellij.openapi.command.UndoConfirmationPolicy | ||||||
| import com.intellij.openapi.components.Service | import com.intellij.openapi.components.Service | ||||||
| @@ -41,8 +39,6 @@ import com.maddyhome.idea.vim.newapi.IjNativeAction | |||||||
| import com.maddyhome.idea.vim.newapi.ij | import com.maddyhome.idea.vim.newapi.ij | ||||||
| import com.maddyhome.idea.vim.newapi.runFromVimKey | import com.maddyhome.idea.vim.newapi.runFromVimKey | ||||||
| import org.jetbrains.annotations.NonNls | import org.jetbrains.annotations.NonNls | ||||||
| import java.awt.Component |  | ||||||
| import javax.swing.JComponent |  | ||||||
| import javax.swing.SwingUtilities | import javax.swing.SwingUtilities | ||||||
|  |  | ||||||
| @Service | @Service | ||||||
| @@ -143,7 +139,7 @@ internal class IjActionExecutor : VimActionExecutor { | |||||||
|       manager.fireAfterActionPerformed(action, event, result!!) |       manager.fireAfterActionPerformed(action, event, result!!) | ||||||
|     } |     } | ||||||
|     if (indexError != null) { |     if (indexError != null) { | ||||||
|       ActionUtil.showDumbModeWarning(project, action, event) |       ActionUtil.showDumbModeWarning(project, event) | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
| @@ -154,44 +150,11 @@ internal class IjActionExecutor : VimActionExecutor { | |||||||
|    * @param context The context to run it in |    * @param context The context to run it in | ||||||
|    */ |    */ | ||||||
|   override fun executeAction(name: @NonNls String, context: ExecutionContext): Boolean { |   override fun executeAction(name: @NonNls String, context: ExecutionContext): Boolean { | ||||||
|     val action = getAction(name, context) |     val aMgr = ActionManager.getInstance() | ||||||
|  |     val action = aMgr.getAction(name) | ||||||
|     return action != null && executeAction(null, IjNativeAction(action), context) |     return action != null && executeAction(null, IjNativeAction(action), context) | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   private fun getAction(name: String, context: ExecutionContext): AnAction? { |  | ||||||
|     val actionManager = ActionManager.getInstance() |  | ||||||
|     val action = actionManager.getAction(name) |  | ||||||
|     if (action !is EmptyAction) return action |  | ||||||
|  |  | ||||||
|     // But if the action is an instance of EmptyAction, the fun begins |  | ||||||
|     var component: Component? = context.ij.getData(PlatformDataKeys.CONTEXT_COMPONENT) ?: return null |  | ||||||
|     while (component != null) { |  | ||||||
|       if (component !is JComponent) { |  | ||||||
|         component = component.parent |  | ||||||
|         continue |  | ||||||
|       } |  | ||||||
|  |  | ||||||
|       val listOfActions = ActionUtil.getActions(component) |  | ||||||
|       if (listOfActions.isEmpty()) { |  | ||||||
|         component = component.getParent() |  | ||||||
|         continue |  | ||||||
|       } |  | ||||||
|  |  | ||||||
|       fun AnAction.getId(): String? { |  | ||||||
|         return actionManager.getId(this) |  | ||||||
|           ?: (shortcutSet as? ProxyShortcutSet)?.actionId |  | ||||||
|       } |  | ||||||
|  |  | ||||||
|       for (action in listOfActions) { |  | ||||||
|         if (action.getId() == name) { |  | ||||||
|           return action |  | ||||||
|         } |  | ||||||
|       } |  | ||||||
|       component = component.getParent() |  | ||||||
|     } |  | ||||||
|     return null |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   override fun executeCommand( |   override fun executeCommand( | ||||||
|     editor: VimEditor?, |     editor: VimEditor?, | ||||||
|     runnable: Runnable, |     runnable: Runnable, | ||||||
|   | |||||||
| @@ -14,7 +14,6 @@ import com.maddyhome.idea.vim.api.injector | |||||||
| import com.maddyhome.idea.vim.api.normalizeVisualColumn | import com.maddyhome.idea.vim.api.normalizeVisualColumn | ||||||
| import com.maddyhome.idea.vim.api.options | import com.maddyhome.idea.vim.api.options | ||||||
| import com.maddyhome.idea.vim.command.CommandFlags | import com.maddyhome.idea.vim.command.CommandFlags | ||||||
| import com.maddyhome.idea.vim.state.VimStateMachine |  | ||||||
| import com.maddyhome.idea.vim.helper.EditorHelper.getApproximateScreenHeight | import com.maddyhome.idea.vim.helper.EditorHelper.getApproximateScreenHeight | ||||||
| import com.maddyhome.idea.vim.helper.EditorHelper.getApproximateScreenWidth | import com.maddyhome.idea.vim.helper.EditorHelper.getApproximateScreenWidth | ||||||
| import com.maddyhome.idea.vim.helper.EditorHelper.getNonNormalizedVisualLineAtBottomOfScreen | import com.maddyhome.idea.vim.helper.EditorHelper.getNonNormalizedVisualLineAtBottomOfScreen | ||||||
| @@ -29,6 +28,7 @@ import com.maddyhome.idea.vim.helper.EditorHelper.scrollVisualLineToBottomOfScre | |||||||
| import com.maddyhome.idea.vim.helper.EditorHelper.scrollVisualLineToMiddleOfScreen | import com.maddyhome.idea.vim.helper.EditorHelper.scrollVisualLineToMiddleOfScreen | ||||||
| import com.maddyhome.idea.vim.helper.EditorHelper.scrollVisualLineToTopOfScreen | import com.maddyhome.idea.vim.helper.EditorHelper.scrollVisualLineToTopOfScreen | ||||||
| import com.maddyhome.idea.vim.newapi.vim | import com.maddyhome.idea.vim.newapi.vim | ||||||
|  | import com.maddyhome.idea.vim.state.VimStateMachine | ||||||
| import kotlin.math.max | import kotlin.math.max | ||||||
| import kotlin.math.min | import kotlin.math.min | ||||||
| import kotlin.math.roundToInt | import kotlin.math.roundToInt | ||||||
| @@ -56,7 +56,7 @@ internal object ScrollViewHelper { | |||||||
|     // that this needs to be replaced as a more or less dumb line for line rewrite. |     // that this needs to be replaced as a more or less dumb line for line rewrite. | ||||||
|     val topLine = getVisualLineAtTopOfScreen(editor) |     val topLine = getVisualLineAtTopOfScreen(editor) | ||||||
|     val bottomLine = getVisualLineAtBottomOfScreen(editor) |     val bottomLine = getVisualLineAtBottomOfScreen(editor) | ||||||
|     val lastLine = vimEditor.getVisualLineCount() - 1 |     val lastLine = vimEditor.getVisualLineCount() + editor.settings.additionalLinesCount | ||||||
|  |  | ||||||
|     // We need the non-normalised value here, so we can handle cases such as so=999 to keep the current line centred |     // We need the non-normalised value here, so we can handle cases such as so=999 to keep the current line centred | ||||||
|     val scrollOffset = injector.options(vimEditor).scrolloff |     val scrollOffset = injector.options(vimEditor).scrolloff | ||||||
|   | |||||||
| @@ -11,16 +11,21 @@ package com.maddyhome.idea.vim.helper | |||||||
| import com.intellij.openapi.actionSystem.DataContext | import com.intellij.openapi.actionSystem.DataContext | ||||||
| import com.intellij.openapi.actionSystem.PlatformDataKeys | import com.intellij.openapi.actionSystem.PlatformDataKeys | ||||||
| import com.intellij.openapi.command.CommandProcessor | import com.intellij.openapi.command.CommandProcessor | ||||||
|  | import com.intellij.openapi.command.impl.UndoManagerImpl | ||||||
| import com.intellij.openapi.command.undo.UndoManager | import com.intellij.openapi.command.undo.UndoManager | ||||||
| import com.intellij.openapi.components.Service | import com.intellij.openapi.components.Service | ||||||
| import com.intellij.openapi.fileEditor.impl.text.TextEditorProvider | import com.intellij.openapi.fileEditor.impl.text.TextEditorProvider | ||||||
|  | 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 | ||||||
| import com.maddyhome.idea.vim.api.injector | import com.maddyhome.idea.vim.api.injector | ||||||
| import com.maddyhome.idea.vim.common.ChangesListener | import com.maddyhome.idea.vim.common.ChangesListener | ||||||
|  | import com.maddyhome.idea.vim.group.IjOptions | ||||||
| import com.maddyhome.idea.vim.listener.SelectionVimListenerSuppressor | import com.maddyhome.idea.vim.listener.SelectionVimListenerSuppressor | ||||||
| import com.maddyhome.idea.vim.newapi.globalIjOptions | import com.maddyhome.idea.vim.newapi.globalIjOptions | ||||||
| import com.maddyhome.idea.vim.newapi.ij | import com.maddyhome.idea.vim.newapi.ij | ||||||
|  | import com.maddyhome.idea.vim.state.mode.SelectionType | ||||||
|  | import com.maddyhome.idea.vim.state.mode.inVisualMode | ||||||
| import com.maddyhome.idea.vim.undo.UndoRedoBase | import com.maddyhome.idea.vim.undo.UndoRedoBase | ||||||
|  |  | ||||||
| /** | /** | ||||||
| @@ -28,6 +33,15 @@ import com.maddyhome.idea.vim.undo.UndoRedoBase | |||||||
|  */ |  */ | ||||||
| @Service | @Service | ||||||
| internal class UndoRedoHelper : UndoRedoBase() { | internal class UndoRedoHelper : UndoRedoBase() { | ||||||
|  |   init { | ||||||
|  |     fun onOldUndoChanged() { | ||||||
|  |       UndoManagerImpl.ourNeverAskUser = !injector.globalIjOptions().oldundo | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     injector.optionGroup.addGlobalOptionChangeListener(IjOptions.oldundo, ::onOldUndoChanged) | ||||||
|  |     onOldUndoChanged() | ||||||
|  |   } | ||||||
|  |  | ||||||
|   override fun undo(editor: VimEditor, context: ExecutionContext): Boolean { |   override fun undo(editor: VimEditor, context: ExecutionContext): Boolean { | ||||||
|     val ijContext = context.context as DataContext |     val ijContext = context.context as DataContext | ||||||
|     val project = PlatformDataKeys.PROJECT.getData(ijContext) ?: return false |     val project = PlatformDataKeys.PROJECT.getData(ijContext) ?: return false | ||||||
| @@ -39,19 +53,20 @@ internal class UndoRedoHelper : UndoRedoBase() { | |||||||
|  |  | ||||||
|       if (injector.globalIjOptions().oldundo) { |       if (injector.globalIjOptions().oldundo) { | ||||||
|         SelectionVimListenerSuppressor.lock().use { undoManager.undo(fileEditor) } |         SelectionVimListenerSuppressor.lock().use { undoManager.undo(fileEditor) } | ||||||
|  |         restoreVisualMode(editor) | ||||||
|       } else { |       } else { | ||||||
|         // TODO refactor me after VIM-308 when restoring selection and caret movement will be ignored by undo |         performUntilFileChanges(editor, { undoManager.isUndoAvailable(fileEditor) }, { undoManager.undo(fileEditor) }) | ||||||
|         editor.runWithChangeTracking { |  | ||||||
|           undoManager.undo(fileEditor) |  | ||||||
|  |  | ||||||
|           // We execute undo one more time if the previous one just restored selection |  | ||||||
|           if (!hasChanges && hasSelection(editor) && undoManager.isUndoAvailable(fileEditor)) { |  | ||||||
|             undoManager.undo(fileEditor) |  | ||||||
|           } |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|  |         editor.carets().forEach { | ||||||
|  |           val ijCaret = it.ij | ||||||
|  |           val hasSelection = ijCaret.hasSelection() | ||||||
|  |           if (hasSelection) { | ||||||
|  |             val selectionStart = ijCaret.selectionStart | ||||||
|             CommandProcessor.getInstance().runUndoTransparentAction { |             CommandProcessor.getInstance().runUndoTransparentAction { | ||||||
|           removeSelections(editor) |               it.ij.removeSelection() | ||||||
|  |               it.ij.moveToOffset(selectionStart) | ||||||
|  |             } | ||||||
|  |           } | ||||||
|         } |         } | ||||||
|       } |       } | ||||||
|  |  | ||||||
| @@ -62,10 +77,6 @@ internal class UndoRedoHelper : UndoRedoBase() { | |||||||
|     return false |     return false | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   private fun hasSelection(editor: VimEditor): Boolean { |  | ||||||
|     return editor.primaryCaret().ij.hasSelection() |  | ||||||
|   } |  | ||||||
|    |  | ||||||
|   override fun redo(editor: VimEditor, context: ExecutionContext): Boolean { |   override fun redo(editor: VimEditor, context: ExecutionContext): Boolean { | ||||||
|     val ijContext = context.context as DataContext |     val ijContext = context.context as DataContext | ||||||
|     val project = PlatformDataKeys.PROJECT.getData(ijContext) ?: return false |     val project = PlatformDataKeys.PROJECT.getData(ijContext) ?: return false | ||||||
| @@ -74,49 +85,23 @@ internal class UndoRedoHelper : UndoRedoBase() { | |||||||
|     if (undoManager.isRedoAvailable(fileEditor)) { |     if (undoManager.isRedoAvailable(fileEditor)) { | ||||||
|       if (injector.globalIjOptions().oldundo) { |       if (injector.globalIjOptions().oldundo) { | ||||||
|         SelectionVimListenerSuppressor.lock().use { undoManager.redo(fileEditor) } |         SelectionVimListenerSuppressor.lock().use { undoManager.redo(fileEditor) } | ||||||
|  |         restoreVisualMode(editor) | ||||||
|       } else { |       } else { | ||||||
|         undoManager.redo(fileEditor) |         performUntilFileChanges(editor, { undoManager.isRedoAvailable(fileEditor) }, { undoManager.redo(fileEditor) }) | ||||||
|         CommandProcessor.getInstance().runUndoTransparentAction { |         CommandProcessor.getInstance().runUndoTransparentAction { | ||||||
|           editor.carets().forEach { it.ij.removeSelection() } |           editor.carets().forEach { it.ij.removeSelection() } | ||||||
|         } |         } | ||||||
|         // TODO refactor me after VIM-308 when restoring selection and caret movement will be ignored by undo |  | ||||||
|         editor.runWithChangeTracking { |  | ||||||
|           undoManager.redo(fileEditor) |  | ||||||
|  |  | ||||||
|           // We execute undo one more time if the previous one just restored selection |  | ||||||
|           if (!hasChanges && hasSelection(editor) && undoManager.isRedoAvailable(fileEditor)) { |  | ||||||
|             undoManager.redo(fileEditor) |  | ||||||
|           } |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         CommandProcessor.getInstance().runUndoTransparentAction { |  | ||||||
|           removeSelections(editor) |  | ||||||
|         } |  | ||||||
|       } |       } | ||||||
|       return true |       return true | ||||||
|     } |     } | ||||||
|     return false |     return false | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   private fun removeSelections(editor: VimEditor) { |   private fun performUntilFileChanges(editor: VimEditor?, check: () -> Boolean, action: Runnable) { | ||||||
|     editor.carets().forEach { |     if (editor == null) return | ||||||
|       val ijCaret = it.ij |     val vimDocument = editor.document | ||||||
|       if (!ijCaret.hasSelection()) return@forEach |  | ||||||
|  |  | ||||||
|       val selectionStart = ijCaret.selectionStart |     val changeListener = object : ChangesListener { | ||||||
|       ijCaret.removeSelection() |  | ||||||
|       ijCaret.moveToOffset(selectionStart) |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   private fun VimEditor.runWithChangeTracking(block: ChangeTracker.() -> Unit) { |  | ||||||
|     val tracker = ChangeTracker(this) |  | ||||||
|     tracker.block() |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   private class ChangeTracker(private val editor: VimEditor) { |  | ||||||
|     private val initialPath = editor.getPath() |  | ||||||
|     private val changeListener = object : ChangesListener { |  | ||||||
|       var hasChanged = false |       var hasChanged = false | ||||||
|  |  | ||||||
|       override fun documentChanged(change: ChangesListener.Change) { |       override fun documentChanged(change: ChangesListener.Change) { | ||||||
| @@ -124,11 +109,32 @@ internal class UndoRedoHelper : UndoRedoBase() { | |||||||
|       } |       } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     init { |     val oldPath = editor.getPath() | ||||||
|       editor.document.addChangeListener(changeListener) |     vimDocument.addChangeListener(changeListener) | ||||||
|  |     while (check() && !changeListener.hasChanged && !ifFilePathChanged(editor, oldPath)) { | ||||||
|  |       action.run() | ||||||
|  |     } | ||||||
|  |     vimDocument.removeChangeListener(changeListener) | ||||||
|   } |   } | ||||||
|  |  | ||||||
|     val hasChanges: Boolean |   private fun ifFilePathChanged(editor: VimEditor, oldPath: String?): Boolean { | ||||||
|       get() = changeListener.hasChanged || initialPath != editor.getPath() |     return editor.getPath() != oldPath | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   private fun restoreVisualMode(editor: VimEditor) { | ||||||
|  |     if (!editor.inVisualMode && editor.getSelectionModel().hasSelection()) { | ||||||
|  |       val detectedMode = VimPlugin.getVisualMotion().autodetectVisualSubmode(editor) | ||||||
|  |        | ||||||
|  |       // Visual block selection is restored into multiple carets, so multi-carets that form a block are always | ||||||
|  |       // identified as visual block mode, leading to false positives. | ||||||
|  |       // Since I use visual block mode much less often than multi-carets, this is a judgment call to never restore | ||||||
|  |       // visual block mode. | ||||||
|  |       val wantedMode = if (detectedMode == SelectionType.BLOCK_WISE) | ||||||
|  |         SelectionType.CHARACTER_WISE | ||||||
|  |       else | ||||||
|  |         detectedMode | ||||||
|  |        | ||||||
|  |       VimPlugin.getVisualMotion().enterVisualMode(editor, wantedMode) | ||||||
|  |     } | ||||||
|   } |   } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -124,6 +124,10 @@ internal var Editor.vimMorePanel: ExOutputPanel? by userData() | |||||||
| internal var Editor.vimExOutput: ExOutputModel? by userData() | internal var Editor.vimExOutput: ExOutputModel? by userData() | ||||||
| internal var Editor.vimTestInputModel: TestInputModel? by userData() | internal var Editor.vimTestInputModel: TestInputModel? by userData() | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * Checks whether a keeping visual mode visual operator action is performed on editor. | ||||||
|  |  */ | ||||||
|  | internal var Editor.vimKeepingVisualOperatorAction: Boolean by userDataOr { false } | ||||||
| internal var Editor.vimChangeActionSwitchMode: Mode? by userData() | internal var Editor.vimChangeActionSwitchMode: Mode? by userData() | ||||||
|  |  | ||||||
| /** | /** | ||||||
|   | |||||||
| @@ -1,32 +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.helper |  | ||||||
|  |  | ||||||
| import com.intellij.ide.plugins.StandalonePluginUpdateChecker |  | ||||||
| import com.intellij.openapi.components.Service |  | ||||||
| import com.intellij.openapi.components.service |  | ||||||
| import com.maddyhome.idea.vim.VimPlugin |  | ||||||
| import com.maddyhome.idea.vim.group.NotificationService |  | ||||||
| import com.maddyhome.idea.vim.icons.VimIcons |  | ||||||
|  |  | ||||||
| @Service(Service.Level.APP) |  | ||||||
| internal class VimStandalonePluginUpdateChecker : StandalonePluginUpdateChecker( |  | ||||||
|   VimPlugin.getPluginId(), |  | ||||||
|   updateTimestampProperty = PROPERTY_NAME, |  | ||||||
|   NotificationService.IDEAVIM_STICKY_GROUP, |  | ||||||
|   VimIcons.IDEAVIM, |  | ||||||
| ) { |  | ||||||
|  |  | ||||||
|   override fun skipUpdateCheck(): Boolean = VimPlugin.isNotEnabled() || "dev" in VimPlugin.getVersion() |  | ||||||
|  |  | ||||||
|   companion object { |  | ||||||
|     private const val PROPERTY_NAME = "ideavim.statistics.timestamp" |  | ||||||
|     val instance: VimStandalonePluginUpdateChecker = service() |  | ||||||
|   } |  | ||||||
| } |  | ||||||
| @@ -1,25 +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.ide |  | ||||||
|  |  | ||||||
| import com.intellij.openapi.extensions.ExtensionPointName |  | ||||||
|  |  | ||||||
| internal val clionEP = ExtensionPointName.create<ClionNovaProvider>("IdeaVIM.clionNovaProvider") |  | ||||||
|  |  | ||||||
| internal interface ClionNovaProvider { |  | ||||||
|   fun isClionNova(): Boolean |  | ||||||
| } |  | ||||||
|  |  | ||||||
| internal class ClionNovaProviderImpl : ClionNovaProvider { |  | ||||||
|   override fun isClionNova(): Boolean = true |  | ||||||
| } |  | ||||||
|  |  | ||||||
| internal fun isClionNova(): Boolean { |  | ||||||
|   return clionEP.extensions.any { it.isClionNova() } |  | ||||||
| } |  | ||||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user