mirror of
https://github.com/chylex/IntelliJ-IdeaVim.git
synced 2025-10-24 18:23:43 +02:00
Compare commits
9 Commits
ea703bce69
...
customized
Author | SHA1 | Date | |
---|---|---|---|
8bcb0d116d
|
|||
d9ae9fa40d
|
|||
c19f88e5c0
|
|||
03f4fb288d
|
|||
ef27579277
|
|||
a8a822b58e
|
|||
9752bd5bbb
|
|||
240d9e0be4
|
|||
328442544d
|
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.1
|
||||||
downloadIdeaSources=true
|
downloadIdeaSources=true
|
||||||
instrumentPluginCode=true
|
instrumentPluginCode=true
|
||||||
version=SNAPSHOT
|
version=chylex-16
|
||||||
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"
|
|
||||||
}
|
|
@@ -28,6 +28,7 @@ import com.maddyhome.idea.vim.api.globalOptions
|
|||||||
import com.maddyhome.idea.vim.api.injector
|
import com.maddyhome.idea.vim.api.injector
|
||||||
import com.maddyhome.idea.vim.group.IjOptionConstants
|
import com.maddyhome.idea.vim.group.IjOptionConstants
|
||||||
import com.maddyhome.idea.vim.group.IjOptions
|
import com.maddyhome.idea.vim.group.IjOptions
|
||||||
|
import com.maddyhome.idea.vim.handler.enableOctopus
|
||||||
import com.maddyhome.idea.vim.handler.isOctopusEnabled
|
import com.maddyhome.idea.vim.handler.isOctopusEnabled
|
||||||
import com.maddyhome.idea.vim.helper.EditorHelper
|
import com.maddyhome.idea.vim.helper.EditorHelper
|
||||||
import com.maddyhome.idea.vim.helper.HandlerInjector
|
import com.maddyhome.idea.vim.helper.HandlerInjector
|
||||||
@@ -54,17 +55,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 +91,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 (isOctopusEnabled(keyStroke, editor)) {
|
if (enableOctopus) {
|
||||||
return ActionEnableStatus.no(
|
if (isOctopusEnabled(keyStroke, editor)) {
|
||||||
"Processing VimShortcutKeyAction for the key that is used in the octopus handler",
|
return ActionEnableStatus.no("Octopus handler is enabled", LogLevel.DEBUG)
|
||||||
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 +160,10 @@ public class VimShortcutKeyAction : AnAction(), DumbAware/*, LightEditCompatible
|
|||||||
return ActionEnableStatus.no("App code template is active", LogLevel.INFO)
|
return ActionEnableStatus.no("App code template is active", LogLevel.INFO)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (keyCode == KeyEvent.VK_LEFT || keyCode == KeyEvent.VK_RIGHT || keyCode == KeyEvent.VK_UP || keyCode == KeyEvent.VK_DOWN || keyCode == KeyEvent.VK_HOME || keyCode == KeyEvent.VK_END) {
|
||||||
|
return ActionEnableStatus.no("Special keys", LogLevel.INFO)
|
||||||
|
}
|
||||||
|
|
||||||
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 +229,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 +239,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 +274,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 +365,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 +381,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,26 +74,19 @@ 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(
|
||||||
mode: Mode,
|
mode: Mode,
|
||||||
editor: VimEditor,
|
editor: VimEditor,
|
||||||
@@ -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
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -8,18 +8,21 @@
|
|||||||
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
|
||||||
import com.maddyhome.idea.vim.api.injector
|
import com.maddyhome.idea.vim.api.injector
|
||||||
import com.maddyhome.idea.vim.api.setChangeMarks
|
import com.maddyhome.idea.vim.api.setChangeMarks
|
||||||
import com.maddyhome.idea.vim.command.MappingMode
|
import com.maddyhome.idea.vim.command.MappingMode
|
||||||
|
import com.maddyhome.idea.vim.state.mode.Mode
|
||||||
import com.maddyhome.idea.vim.command.OperatorArguments
|
import com.maddyhome.idea.vim.command.OperatorArguments
|
||||||
|
import com.maddyhome.idea.vim.state.mode.SelectionType
|
||||||
|
import com.maddyhome.idea.vim.state.mode.selectionType
|
||||||
import com.maddyhome.idea.vim.common.TextRange
|
import com.maddyhome.idea.vim.common.TextRange
|
||||||
import com.maddyhome.idea.vim.extension.ExtensionHandler
|
import com.maddyhome.idea.vim.extension.ExtensionHandler
|
||||||
import com.maddyhome.idea.vim.extension.VimExtension
|
import com.maddyhome.idea.vim.extension.VimExtension
|
||||||
@@ -31,15 +34,15 @@ import com.maddyhome.idea.vim.extension.VimExtensionFacade.putExtensionHandlerMa
|
|||||||
import com.maddyhome.idea.vim.extension.VimExtensionFacade.putKeyMappingIfMissing
|
import com.maddyhome.idea.vim.extension.VimExtensionFacade.putKeyMappingIfMissing
|
||||||
import com.maddyhome.idea.vim.extension.VimExtensionFacade.setOperatorFunction
|
import com.maddyhome.idea.vim.extension.VimExtensionFacade.setOperatorFunction
|
||||||
import com.maddyhome.idea.vim.extension.VimExtensionFacade.setRegisterForCaret
|
import com.maddyhome.idea.vim.extension.VimExtensionFacade.setRegisterForCaret
|
||||||
|
import com.maddyhome.idea.vim.state.mode.mode
|
||||||
|
import com.maddyhome.idea.vim.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
|
||||||
import com.maddyhome.idea.vim.put.PutData
|
import com.maddyhome.idea.vim.put.PutData
|
||||||
import com.maddyhome.idea.vim.state.mode.Mode
|
|
||||||
import com.maddyhome.idea.vim.state.mode.SelectionType
|
|
||||||
import com.maddyhome.idea.vim.state.mode.mode
|
|
||||||
import com.maddyhome.idea.vim.state.mode.selectionType
|
|
||||||
import org.jetbrains.annotations.NonNls
|
import org.jetbrains.annotations.NonNls
|
||||||
import java.awt.event.KeyEvent
|
import java.awt.event.KeyEvent
|
||||||
import javax.swing.KeyStroke
|
import javax.swing.KeyStroke
|
||||||
@@ -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)) // TODO
|
||||||
executeNormalWithoutMapping(injector.parser.parseKeys("g@"), editor.ij)
|
executeNormalWithoutMapping(injector.parser.parseKeys("g@"), editor.ij)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -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).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,51 @@ 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) : 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()
|
||||||
// Jump back to start
|
if (supportsMultipleCursors) {
|
||||||
executeNormalWithoutMapping(injector.parser.parseKeys("`["), ijEditor)
|
editor.runWithEveryCaretAndRestore {
|
||||||
|
applyOnce(editor, change, pair)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
applyOnce(editor, change, pair)
|
||||||
|
// Jump back to start
|
||||||
|
executeNormalWithoutMapping(injector.parser.parseKeys("`["), editor)
|
||||||
|
}
|
||||||
|
}
|
||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private fun applyOnce(editor: Editor, change: VimChangeGroup, pair: Pair<String, String>) {
|
||||||
|
// XXX: Will it work with line-wise or block-wise selections?
|
||||||
|
val primaryCaret = editor.caretModel.primaryCaret
|
||||||
|
val range = getSurroundRange(primaryCaret.vim)
|
||||||
|
if (range != null) {
|
||||||
|
change.insertText(IjVimEditor(editor), IjVimCaret(primaryCaret), range.startOffset, pair.first)
|
||||||
|
change.insertText(
|
||||||
|
IjVimEditor(editor),
|
||||||
|
IjVimCaret(primaryCaret),
|
||||||
|
range.endOffset + pair.first.length,
|
||||||
|
pair.second
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
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,101 +310,96 @@ 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(
|
||||||
|
'b' to ("(" to ")"),
|
||||||
|
'(' to ("( " to " )"),
|
||||||
|
')' to ("(" to ")"),
|
||||||
|
'B' to ("{" to "}"),
|
||||||
|
'{' to ("{ " to " }"),
|
||||||
|
'}' to ("{" to "}"),
|
||||||
|
'r' to ("[" to "]"),
|
||||||
|
'[' to ("[ " to " ]"),
|
||||||
|
']' to ("[" to "]"),
|
||||||
|
'a' to ("<" to ">"),
|
||||||
|
'>' to ("<" to ">"),
|
||||||
|
's' to (" " to ""),
|
||||||
|
)
|
||||||
|
|
||||||
private val SURROUND_PAIRS = mapOf(
|
private fun getSurroundPair(c: Char): Pair<String, String>? = if (c in SURROUND_PAIRS) {
|
||||||
'b' to ("(" to ")"),
|
SURROUND_PAIRS[c]
|
||||||
'(' to ("( " to " )"),
|
} else if (!c.isLetter()) {
|
||||||
')' to ("(" to ")"),
|
val s = c.toString()
|
||||||
'B' to ("{" to "}"),
|
s to s
|
||||||
'{' to ("{ " to " }"),
|
|
||||||
'}' to ("{" to "}"),
|
|
||||||
'r' to ("[" to "]"),
|
|
||||||
'[' to ("[ " to " ]"),
|
|
||||||
']' to ("[" to "]"),
|
|
||||||
'a' to ("<" to ">"),
|
|
||||||
'>' to ("<" to ">"),
|
|
||||||
's' to (" " to ""),
|
|
||||||
)
|
|
||||||
|
|
||||||
private fun getSurroundPair(c: Char): Pair<String, String>? = if (c in SURROUND_PAIRS) {
|
|
||||||
SURROUND_PAIRS[c]
|
|
||||||
} else if (!c.isLetter()) {
|
|
||||||
val s = c.toString()
|
|
||||||
s to s
|
|
||||||
} else {
|
|
||||||
null
|
|
||||||
}
|
|
||||||
|
|
||||||
private fun inputTagPair(editor: Editor): Pair<String, String>? {
|
|
||||||
val tagInput = inputString(editor, "<", '>')
|
|
||||||
val matcher = tagNameAndAttributesCapturePattern.matcher(tagInput)
|
|
||||||
return if (matcher.find()) {
|
|
||||||
val tagName = matcher.group(1)
|
|
||||||
val tagAttributes = matcher.group(2)
|
|
||||||
"<$tagName$tagAttributes>" to "</$tagName>"
|
|
||||||
} else {
|
|
||||||
null
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private fun inputFunctionName(
|
|
||||||
editor: Editor,
|
|
||||||
withInternalSpaces: Boolean,
|
|
||||||
): Pair<String, String>? {
|
|
||||||
val functionNameInput = inputString(editor, "function: ", null)
|
|
||||||
if (functionNameInput.isEmpty()) return null
|
|
||||||
return if (withInternalSpaces) "$functionNameInput( " to " )" else "$functionNameInput(" to ")"
|
|
||||||
}
|
|
||||||
|
|
||||||
private fun getOrInputPair(c: Char, editor: Editor): Pair<String, String>? = when (c) {
|
|
||||||
'<', 't' -> inputTagPair(editor)
|
|
||||||
'f' -> inputFunctionName(editor, false)
|
|
||||||
'F' -> inputFunctionName(editor, true)
|
|
||||||
else -> getSurroundPair(c)
|
|
||||||
}
|
|
||||||
|
|
||||||
private fun getChar(editor: Editor): Char {
|
|
||||||
val key = inputKeyStroke(editor)
|
|
||||||
val keyChar = key.keyChar
|
|
||||||
val res = if (keyChar == KeyEvent.CHAR_UNDEFINED || keyChar.code == KeyEvent.VK_ESCAPE) {
|
|
||||||
0.toChar()
|
|
||||||
} else {
|
|
||||||
keyChar
|
|
||||||
}
|
|
||||||
LOG.trace("getChar: $res")
|
|
||||||
return res
|
|
||||||
}
|
|
||||||
|
|
||||||
private fun performSurround(pair: Pair<String, String>, range: TextRange, caret: VimCaret, tagsOnNewLines: Boolean = false) {
|
|
||||||
runWriteAction {
|
|
||||||
val editor = caret.editor
|
|
||||||
val change = VimPlugin.getChange()
|
|
||||||
val leftSurround = pair.first + if (tagsOnNewLines) "\n" else ""
|
|
||||||
|
|
||||||
val isEOF = range.endOffset == editor.text().length
|
|
||||||
val hasNewLine = editor.endsWithNewLine()
|
|
||||||
val rightSurround = if (tagsOnNewLines) {
|
|
||||||
if (isEOF && !hasNewLine) {
|
|
||||||
"\n" + pair.second
|
|
||||||
} else {
|
|
||||||
pair.second + "\n"
|
|
||||||
}
|
|
||||||
} else {
|
} else {
|
||||||
pair.second
|
null
|
||||||
}
|
}
|
||||||
|
|
||||||
change.insertText(editor, caret, range.startOffset, leftSurround)
|
private fun inputTagPair(editor: Editor): Pair<String, String>? {
|
||||||
change.insertText(editor, caret, range.endOffset + leftSurround.length, rightSurround)
|
val tagInput = inputString(editor, "<", '>')
|
||||||
injector.markService.setChangeMarks(
|
val matcher = tagNameAndAttributesCapturePattern.matcher(tagInput)
|
||||||
caret,
|
return if (matcher.find()) {
|
||||||
TextRange(range.startOffset, range.endOffset + leftSurround.length + rightSurround.length)
|
val tagName = matcher.group(1)
|
||||||
)
|
val tagAttributes = matcher.group(2)
|
||||||
|
"<$tagName$tagAttributes>" to "</$tagName>"
|
||||||
|
} else {
|
||||||
|
null
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun inputFunctionName(
|
||||||
|
editor: Editor,
|
||||||
|
withInternalSpaces: Boolean,
|
||||||
|
): Pair<String, String>? {
|
||||||
|
val functionNameInput = inputString(editor, "function: ", null)
|
||||||
|
if (functionNameInput.isEmpty()) return null
|
||||||
|
return if (withInternalSpaces) "$functionNameInput( " to " )" else "$functionNameInput(" to ")"
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun getOrInputPair(c: Char, editor: Editor): Pair<String, String>? = when (c) {
|
||||||
|
'<', 't' -> inputTagPair(editor)
|
||||||
|
'f' -> inputFunctionName(editor, false)
|
||||||
|
'F' -> inputFunctionName(editor, true)
|
||||||
|
else -> getSurroundPair(c)
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun getChar(editor: Editor): Char {
|
||||||
|
val key = inputKeyStroke(editor)
|
||||||
|
val keyChar = key.keyChar
|
||||||
|
return if (keyChar == KeyEvent.CHAR_UNDEFINED || keyChar.code == KeyEvent.VK_ESCAPE) {
|
||||||
|
0.toChar()
|
||||||
|
} else {
|
||||||
|
keyChar
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun performSurround(pair: Pair<String, String>, range: TextRange, caret: VimCaret, tagsOnNewLines: Boolean = false) {
|
||||||
|
runWriteAction {
|
||||||
|
val editor = caret.editor
|
||||||
|
val change = VimPlugin.getChange()
|
||||||
|
val leftSurround = pair.first + if (tagsOnNewLines) "\n" else ""
|
||||||
|
|
||||||
|
val isEOF = range.endOffset == editor.text().length
|
||||||
|
val hasNewLine = editor.endsWithNewLine()
|
||||||
|
val rightSurround = if (tagsOnNewLines) {
|
||||||
|
if (isEOF && !hasNewLine) {
|
||||||
|
"\n" + pair.second
|
||||||
|
} else {
|
||||||
|
pair.second + "\n"
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
pair.second
|
||||||
|
}
|
||||||
|
|
||||||
|
change.insertText(editor, caret, range.startOffset, leftSurround)
|
||||||
|
change.insertText(editor, caret, range.endOffset + leftSurround.length, rightSurround)
|
||||||
|
injector.markService.setChangeMarks(caret, TextRange(range.startOffset, range.endOffset + leftSurround.length + rightSurround.length))
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -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,33 +61,22 @@ 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) {
|
myPotemkinProgress.fraction = (i + 1).toDouble() / total
|
||||||
|
while (keyStack.hasStroke()) {
|
||||||
|
val key = keyStack.feedStroke()
|
||||||
try {
|
try {
|
||||||
myPotemkinProgress.fraction = (i + 1).toDouble() / total
|
myPotemkinProgress.checkCanceled()
|
||||||
while (keyStack.hasStroke()) {
|
} catch (e: ProcessCanceledException) {
|
||||||
val key = keyStack.feedStroke()
|
return@runnable
|
||||||
try {
|
|
||||||
myPotemkinProgress.checkCanceled()
|
|
||||||
} catch (e: ProcessCanceledException) {
|
|
||||||
return@runnable
|
|
||||||
}
|
|
||||||
ProgressManager.getInstance().executeNonCancelableSection {
|
|
||||||
// Prevent autocompletion during macros.
|
|
||||||
// See https://github.com/JetBrains/ideavim/pull/772 for details
|
|
||||||
CompletionServiceImpl.setCompletionPhase(CompletionPhase.NoCompletion)
|
|
||||||
getInstance().handleKey(editor, key, context)
|
|
||||||
}
|
|
||||||
if (injector.messages.isError()) return@runnable
|
|
||||||
}
|
|
||||||
} finally {
|
|
||||||
keyStack.resetFirst()
|
|
||||||
}
|
}
|
||||||
|
ProgressManager.getInstance().executeNonCancelableSection { getInstance().handleKey(editor, key, context) }
|
||||||
|
if (injector.messages.isError()) return@runnable
|
||||||
}
|
}
|
||||||
} finally {
|
keyStack.resetFirst()
|
||||||
keyStack.removeFirst()
|
|
||||||
}
|
}
|
||||||
|
keyStack.removeFirst()
|
||||||
}
|
}
|
||||||
|
|
||||||
if (isInternalMacro) {
|
if (isInternalMacro) {
|
||||||
|
@@ -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,13 +459,11 @@ 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) {
|
vimEditor.exitVisualMode()
|
||||||
vimEditor.exitVisualMode()
|
KeyHandler.getInstance().reset(vimEditor)
|
||||||
KeyHandler.getInstance().reset(vimEditor)
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -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,51 +40,39 @@ 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) {
|
for (jump in jumps) {
|
||||||
val projectElement = Element("project").setAttribute("id", project)
|
val jumpElem = Element("jump")
|
||||||
for (jump in jumps) {
|
jumpElem.setAttribute("line", jump.line.toString())
|
||||||
val jumpElem = Element("jump")
|
jumpElem.setAttribute("column", jump.col.toString())
|
||||||
jumpElem.setAttribute("line", jump.line.toString())
|
jumpElem.setAttribute("filename", StringUtil.notNullize(jump.filepath))
|
||||||
jumpElem.setAttribute("column", jump.col.toString())
|
jumpsElem.addContent(jumpElem)
|
||||||
jumpElem.setAttribute("filename", StringUtil.notNullize(jump.filepath))
|
if (logger.isDebug()) {
|
||||||
projectElement.addContent(jumpElem)
|
logger.debug("saved jump = $jump")
|
||||||
if (logger.isDebug()) {
|
|
||||||
logger.debug("saved jump = $jump")
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
projectsElem.addContent(projectElement)
|
|
||||||
}
|
}
|
||||||
return projectsElem
|
return jumpsElem
|
||||||
}
|
}
|
||||||
|
|
||||||
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 jump = Jump(
|
||||||
val jumpElements = projectElement.getChildren("jump")
|
Integer.parseInt(jumpElement.getAttributeValue("line")),
|
||||||
for (jumpElement in jumpElements) {
|
Integer.parseInt(jumpElement.getAttributeValue("column")),
|
||||||
val jump = Jump(
|
jumpElement.getAttributeValue("filename"),
|
||||||
Integer.parseInt(jumpElement.getAttributeValue("line")),
|
)
|
||||||
Integer.parseInt(jumpElement.getAttributeValue("column")),
|
jumps.add(jump)
|
||||||
jumpElement.getAttributeValue("filename"),
|
}
|
||||||
)
|
|
||||||
jumps.add(jump)
|
if (logger.isDebug()) {
|
||||||
}
|
logger.debug("jumps=$jumps")
|
||||||
if (logger.isDebug()) {
|
|
||||||
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)
|
executeHandler(editor, caret, dataContext)
|
||||||
val executionHandler = {
|
|
||||||
try {
|
|
||||||
(dataContext as? UserDataHolder)?.putUserData(commandContinuation, nextHandler)
|
|
||||||
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 {
|
|
||||||
executionHandler()
|
|
||||||
}
|
|
||||||
} else {
|
} else {
|
||||||
nextHandler?.execute(editor, caret, dataContext)
|
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
|
||||||
|
@@ -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,
|
||||||
|
@@ -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,20 +53,21 @@ 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
|
editor.carets().forEach {
|
||||||
if (!hasChanges && hasSelection(editor) && undoManager.isUndoAvailable(fileEditor)) {
|
val ijCaret = it.ij
|
||||||
undoManager.undo(fileEditor)
|
val hasSelection = ijCaret.hasSelection()
|
||||||
|
if (hasSelection) {
|
||||||
|
val selectionStart = ijCaret.selectionStart
|
||||||
|
CommandProcessor.getInstance().runUndoTransparentAction {
|
||||||
|
it.ij.removeSelection()
|
||||||
|
it.ij.moveToOffset(selectionStart)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
CommandProcessor.getInstance().runUndoTransparentAction {
|
|
||||||
removeSelections(editor)
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
scrollingModel.flushViewportChanges()
|
scrollingModel.flushViewportChanges()
|
||||||
@@ -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() }
|
|
||||||
}
|
|
@@ -40,7 +40,7 @@ internal object AppCodeTemplates {
|
|||||||
private var editor: Editor? = null
|
private var editor: Editor? = null
|
||||||
|
|
||||||
override fun beforeActionPerformed(action: AnAction, event: AnActionEvent) {
|
override fun beforeActionPerformed(action: AnAction, event: AnActionEvent) {
|
||||||
if (VimPlugin.isNotEnabled()) return
|
if (!VimPlugin.isEnabled()) return
|
||||||
|
|
||||||
val hostEditor = event.dataContext.getData(CommonDataKeys.HOST_EDITOR)
|
val hostEditor = event.dataContext.getData(CommonDataKeys.HOST_EDITOR)
|
||||||
if (hostEditor != null) {
|
if (hostEditor != null) {
|
||||||
@@ -49,7 +49,7 @@ internal object AppCodeTemplates {
|
|||||||
}
|
}
|
||||||
|
|
||||||
override fun afterActionPerformed(action: AnAction, event: AnActionEvent, result: AnActionResult) {
|
override fun afterActionPerformed(action: AnAction, event: AnActionEvent, result: AnActionResult) {
|
||||||
if (VimPlugin.isNotEnabled()) return
|
if (!VimPlugin.isEnabled()) return
|
||||||
|
|
||||||
if (ActionManager.getInstance().getId(action) == IdeActions.ACTION_CHOOSE_LOOKUP_ITEM) {
|
if (ActionManager.getInstance().getId(action) == IdeActions.ACTION_CHOOSE_LOOKUP_ITEM) {
|
||||||
val myEditor = editor
|
val myEditor = editor
|
||||||
|
@@ -59,7 +59,7 @@ internal object IdeaSpecifics {
|
|||||||
private var completionPrevDocumentLength: Int? = null
|
private var completionPrevDocumentLength: Int? = null
|
||||||
private var completionPrevDocumentOffset: Int? = null
|
private var completionPrevDocumentOffset: Int? = null
|
||||||
override fun beforeActionPerformed(action: AnAction, event: AnActionEvent) {
|
override fun beforeActionPerformed(action: AnAction, event: AnActionEvent) {
|
||||||
if (VimPlugin.isNotEnabled()) return
|
if (!VimPlugin.isEnabled()) return
|
||||||
|
|
||||||
val hostEditor = event.dataContext.getData(CommonDataKeys.HOST_EDITOR)
|
val hostEditor = event.dataContext.getData(CommonDataKeys.HOST_EDITOR)
|
||||||
if (hostEditor != null) {
|
if (hostEditor != null) {
|
||||||
@@ -92,7 +92,7 @@ internal object IdeaSpecifics {
|
|||||||
}
|
}
|
||||||
|
|
||||||
override fun afterActionPerformed(action: AnAction, event: AnActionEvent, result: AnActionResult) {
|
override fun afterActionPerformed(action: AnAction, event: AnActionEvent, result: AnActionResult) {
|
||||||
if (VimPlugin.isNotEnabled()) return
|
if (!VimPlugin.isEnabled()) return
|
||||||
|
|
||||||
val editor = editor
|
val editor = editor
|
||||||
if (editor != null && action is ChooseItemAction && editor.vimStateMachine?.isRecording == true) {
|
if (editor != null && action is ChooseItemAction && editor.vimStateMachine?.isRecording == true) {
|
||||||
@@ -138,7 +138,7 @@ internal object IdeaSpecifics {
|
|||||||
//region Enter insert mode for surround templates without selection
|
//region Enter insert mode for surround templates without selection
|
||||||
class VimTemplateManagerListener : TemplateManagerListener {
|
class VimTemplateManagerListener : TemplateManagerListener {
|
||||||
override fun templateStarted(state: TemplateState) {
|
override fun templateStarted(state: TemplateState) {
|
||||||
if (VimPlugin.isNotEnabled()) return
|
if (!VimPlugin.isEnabled()) return
|
||||||
val editor = state.editor ?: return
|
val editor = state.editor ?: return
|
||||||
|
|
||||||
state.addTemplateStateListener(object : TemplateEditingAdapter() {
|
state.addTemplateStateListener(object : TemplateEditingAdapter() {
|
||||||
@@ -176,7 +176,7 @@ internal object IdeaSpecifics {
|
|||||||
//region Register shortcuts for lookup and perform partial reset
|
//region Register shortcuts for lookup and perform partial reset
|
||||||
class LookupTopicListener : LookupManagerListener {
|
class LookupTopicListener : LookupManagerListener {
|
||||||
override fun activeLookupChanged(oldLookup: Lookup?, newLookup: Lookup?) {
|
override fun activeLookupChanged(oldLookup: Lookup?, newLookup: Lookup?) {
|
||||||
if (VimPlugin.isNotEnabled()) return
|
if (!VimPlugin.isEnabled()) return
|
||||||
|
|
||||||
// Lookup opened
|
// Lookup opened
|
||||||
if (oldLookup == null && newLookup is LookupImpl) {
|
if (oldLookup == null && newLookup is LookupImpl) {
|
||||||
@@ -199,7 +199,7 @@ internal object IdeaSpecifics {
|
|||||||
//region Hide Vim search highlights when showing IntelliJ search results
|
//region Hide Vim search highlights when showing IntelliJ search results
|
||||||
class VimFindModelListener : FindModelListener {
|
class VimFindModelListener : FindModelListener {
|
||||||
override fun findNextModelChanged() {
|
override fun findNextModelChanged() {
|
||||||
if (VimPlugin.isNotEnabled()) return
|
if (!VimPlugin.isEnabled()) return
|
||||||
VimPlugin.getSearch().clearSearchHighlight()
|
VimPlugin.getSearch().clearSearchHighlight()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -27,7 +27,7 @@ internal class RiderActionListener : AnActionListener {
|
|||||||
|
|
||||||
private var editor: Editor? = null
|
private var editor: Editor? = null
|
||||||
override fun beforeActionPerformed(action: AnAction, event: AnActionEvent) {
|
override fun beforeActionPerformed(action: AnAction, event: AnActionEvent) {
|
||||||
if (VimPlugin.isNotEnabled()) return
|
if (!VimPlugin.isEnabled()) return
|
||||||
|
|
||||||
val hostEditor = event.dataContext.getData(CommonDataKeys.HOST_EDITOR)
|
val hostEditor = event.dataContext.getData(CommonDataKeys.HOST_EDITOR)
|
||||||
if (hostEditor != null) {
|
if (hostEditor != null) {
|
||||||
@@ -36,7 +36,7 @@ internal class RiderActionListener : AnActionListener {
|
|||||||
}
|
}
|
||||||
|
|
||||||
override fun afterActionPerformed(action: AnAction, event: AnActionEvent, result: AnActionResult) {
|
override fun afterActionPerformed(action: AnAction, event: AnActionEvent, result: AnActionResult) {
|
||||||
if (VimPlugin.isNotEnabled()) return
|
if (!VimPlugin.isEnabled()) return
|
||||||
|
|
||||||
//region Extend Selection for Rider
|
//region Extend Selection for Rider
|
||||||
when (ActionManager.getInstance().getId(action)) {
|
when (ActionManager.getInstance().getId(action)) {
|
||||||
|
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user