mirror of
https://github.com/chylex/IntelliJ-IdeaVim.git
synced 2025-10-24 18:23:43 +02:00
Compare commits
12 Commits
macro-code
...
f1323139b4
Author | SHA1 | Date | |
---|---|---|---|
f1323139b4
|
|||
df4aa59310
|
|||
7a3bb5b2d7
|
|||
c4b05957fc
|
|||
db83b89931
|
|||
ce27c3e5ba
|
|||
31358bc983
|
|||
f7e1c9c837
|
|||
c46109caa3
|
|||
9f7ca83306
|
|||
b5761f20d2
|
|||
305c6d2bf9
|
@@ -1,220 +0,0 @@
|
||||
# Changelog Maintenance Instructions
|
||||
|
||||
## Historical Context
|
||||
|
||||
- The changelog was actively maintained until version 2.9.0
|
||||
- There's a gap from 2.10.0 through 2.27.0 where changelog wasn't maintained
|
||||
- We're resuming changelog maintenance from version 2.28.0 onwards
|
||||
- Between 2.9.0 and 2.28.0, include this note: **"Changelog was not maintained for versions 2.10.0 through 2.27.0"**
|
||||
|
||||
## Changelog Structure
|
||||
|
||||
### [To Be Released] Section
|
||||
- All unreleased changes from master branch go here
|
||||
- When a release is made, this section becomes the new version section
|
||||
- Create a new empty `[To Be Released]` section after each release
|
||||
|
||||
### Version Entry Format
|
||||
```
|
||||
## 2.28.0, 2024-MM-DD
|
||||
|
||||
### Features:
|
||||
* Feature description without ticket number
|
||||
* `CommandName` action can be used... | [VIM-XXXX](https://youtrack.jetbrains.com/issue/VIM-XXXX)
|
||||
|
||||
### Fixes:
|
||||
* [VIM-XXXX](https://youtrack.jetbrains.com/issue/VIM-XXXX) Bug fix description
|
||||
|
||||
### Changes:
|
||||
* Other changes
|
||||
```
|
||||
|
||||
## How to Gather Information
|
||||
|
||||
### 1. Check Current State
|
||||
- Read CHANGES.md to find the last documented version
|
||||
- **Important**: Only read the top portion of CHANGES.md (it's a large file)
|
||||
- Focus on the `[To Be Released]` section and recent versions
|
||||
- Note the date of the last entry
|
||||
|
||||
### 2. Find Releases
|
||||
- Use `git tag --list --sort=-version:refname` to see all version tags
|
||||
- Tags like `2.27.0`, `2.27.1` indicate releases
|
||||
- Note: Patch releases (x.x.1, x.x.2) might be on separate branches
|
||||
- Release dates available at: https://plugins.jetbrains.com/plugin/164-ideavim/versions
|
||||
|
||||
### 3. Review Changes
|
||||
```bash
|
||||
# Get commits since last documented version
|
||||
git log --oneline --since="YYYY-MM-DD" --first-parent master
|
||||
|
||||
# Get merged PRs
|
||||
gh pr list --state merged --limit 100 --json number,title,author,mergedAt
|
||||
|
||||
# Check specific release commits
|
||||
git log --oneline <previous-tag>..<new-tag>
|
||||
```
|
||||
|
||||
**Important**: Don't just read commit messages - examine the actual changes:
|
||||
- Use `git show <commit-hash>` to see the full commit content
|
||||
- Look at modified test files to find specific examples of fixed commands
|
||||
- Check the actual code changes to understand what was really fixed or added
|
||||
- Tests often contain the best examples for changelog entries (e.g., exact commands that now work)
|
||||
|
||||
### 4. What to Include
|
||||
- **Features**: New functionality with [VIM-XXXX] ticket numbers if available
|
||||
- **Bug Fixes**: Fixed issues with [VIM-XXXX] ticket references
|
||||
- **Breaking Changes**: Any backwards-incompatible changes
|
||||
- **Deprecations**: Features marked for future removal
|
||||
- **Merged PRs**: Reference significant PRs like "Implement vim-surround (#123)"
|
||||
- Note: PRs have their own inclusion rules - see "Merged PRs Special Rules" section below
|
||||
|
||||
### 5. What to Exclude
|
||||
- Dependabot PRs (author: dependabot[bot])
|
||||
- Claude-generated PRs (check PR author/title)
|
||||
- Internal refactoring with no user impact
|
||||
- Documentation-only changes (unless significant)
|
||||
- Test-only changes
|
||||
- **API module changes** (while in experimental status) - Do not log changes to the `api` module as it's currently experimental
|
||||
- Note: This exclusion should be removed once the API status is no longer experimental
|
||||
- **Internal code changes** - Do not log coding changes that users cannot see or experience
|
||||
- Refactoring, code cleanup, internal architecture changes
|
||||
- Performance optimizations (unless they fix a noticeable user issue)
|
||||
- Remember: The changelog is for users, not developers
|
||||
|
||||
## Writing Style
|
||||
|
||||
- **Be concise**: One line per change when possible
|
||||
- **User-focused**: Describe what changed from user's perspective
|
||||
- Write for end users, not developers
|
||||
- Focus on visible behavior changes, new commands, fixed issues users experience
|
||||
- Avoid technical implementation details
|
||||
- **Include examples** when helpful:
|
||||
- For fixes: Show the command/operation that now works correctly
|
||||
- For features: Demonstrate the new commands or functionality
|
||||
- Good example: "Fixed `ci"` command in empty strings" or "Added support for `gn` text object"
|
||||
- Bad examples (too vague, unclear what was broken):
|
||||
- "Fixed count validation in text objects"
|
||||
- "Fixed inlay offset calculations"
|
||||
- Better: Specify the actual case - "Fixed `3daw` deleting wrong number of words" or "Fixed cursor position with inlay hints in `f` motion"
|
||||
- **If you can't determine the specific case from tests/code, omit the entry rather than leave it unclear**
|
||||
- **Add helpful links** for context:
|
||||
- When mentioning IntelliJ features, search for official JetBrains documentation or blog posts
|
||||
- When referencing Vim commands, link to Vim documentation if helpful
|
||||
- Example: "Added support for [Next Edit Suggestion](https://blog.jetbrains.com/ai/2025/08/introducing-next-edit-suggestions-in-jetbrains-ai-assistant/)"
|
||||
- Use web search to find the most relevant official sources
|
||||
- **Include references**: Add [VIM-XXXX] for YouTrack tickets, (#XXX) for PRs
|
||||
- **Group logically**: Features, Fixes, Changes, Merged PRs
|
||||
- **No duplication**: Each change appears in exactly ONE subsection - don't repeat items across categories
|
||||
- **Use consistent tense**: Past tense for completed work
|
||||
|
||||
## Examples of Good Entries
|
||||
|
||||
```
|
||||
### Features:
|
||||
* Added support for `gn` text object - select next match with `gn`, change with `cgn`
|
||||
* Implemented `:tabmove` command - use `:tabmove +1` or `:tabmove -1` to reorder tabs
|
||||
* Support for `z=` to show spelling suggestions
|
||||
* Added integration with [Next Edit Suggestion](https://blog.jetbrains.com/ai/2025/08/introducing-next-edit-suggestions-in-jetbrains-ai-assistant/) feature
|
||||
* Support for [multiple cursors](https://www.jetbrains.com/help/idea/multicursor.html) in visual mode
|
||||
|
||||
### Fixes:
|
||||
* [VIM-3456](https://youtrack.jetbrains.com/issue/VIM-3456) Fixed cursor position after undo in visual mode
|
||||
* [VIM-3458](https://youtrack.jetbrains.com/issue/VIM-3458) Fixed `ci"` command now works correctly in empty strings
|
||||
* [VIM-3260](https://youtrack.jetbrains.com/issue/VIM-3260) Fixed `G` command at file end with count
|
||||
* [VIM-3180](https://youtrack.jetbrains.com/issue/VIM-3180) Fixed `vib` and `viB` selection in nested blocks
|
||||
|
||||
### Merged PRs:
|
||||
* [805](https://github.com/JetBrains/ideavim/pull/805) by [chylex](https://github.com/chylex): VIM-3238 Fix recording a macro that replays another macro
|
||||
```
|
||||
|
||||
## IMPORTANT Format Notes
|
||||
|
||||
### For Fixes:
|
||||
Always put the ticket link FIRST, then the description:
|
||||
```
|
||||
* [VIM-XXXX](https://youtrack.jetbrains.com/issue/VIM-XXXX) Description of what was fixed
|
||||
```
|
||||
|
||||
### For Features:
|
||||
- Without ticket: Just the description
|
||||
- With ticket: Can use either format:
|
||||
- Description with pipe: `* Feature description | [VIM-XXXX](https://youtrack.jetbrains.com/issue/VIM-XXXX)`
|
||||
- Link first (like fixes): `* [VIM-XXXX](https://youtrack.jetbrains.com/issue/VIM-XXXX) Feature description`
|
||||
|
||||
### Avoid Duplication:
|
||||
- **Each change should appear in only ONE subsection**
|
||||
- If a feature is listed in Features, don't repeat it in Fixes
|
||||
- If a bug fix is in Fixes, don't list it again elsewhere
|
||||
- Choose the most appropriate category for each change
|
||||
|
||||
### Merged PRs Special Rules:
|
||||
- **Different criteria than other sections**: The exclusion rules for Features/Fixes don't apply here
|
||||
- **Include PRs from external contributors** even if they're internal changes or refactoring
|
||||
- **List significant community contributions** regardless of whether they're user-visible
|
||||
- **Format**: PR number, author, and brief description
|
||||
- **Use PR title as-is**: Take the description directly from the PR title, don't regenerate or rewrite it
|
||||
- **Purpose**: Acknowledge community contributions and provide PR tracking
|
||||
- The "user-visible only" rule does NOT apply to this section
|
||||
|
||||
## Process
|
||||
|
||||
1. Read the current CHANGES.md (only the top portion - focus on `[To Be Released]` and recent versions)
|
||||
2. Check previous changelog PRs from GitHub:
|
||||
- Review the last few changelog update PRs (use `gh pr list --search "Update changelog" --state all --limit 5`)
|
||||
- **Read the PR comments**: Use `gh pr view <PR_NUMBER> --comments` to check for specific instructions
|
||||
- Look for any comments or instructions about what NOT to log this time
|
||||
- Previous PRs may contain specific exclusions or special handling instructions
|
||||
- Pay attention to review feedback that might indicate what to avoid in future updates
|
||||
3. Check git tags for any undocumented releases
|
||||
4. Review commits and PRs since last entry
|
||||
5. Group changes by release or under [To Be Released]
|
||||
6. Update CHANGES.md maintaining existing format
|
||||
7. Update the `changeNotes` section in `build.gradle.kts` (see detailed instructions below)
|
||||
8. Create a PR only if there are changes to document:
|
||||
- Title format: "Update changelog: <super short summary>"
|
||||
- Example: "Update changelog: Add gn text object, fix visual mode issues"
|
||||
- Body: Brief summary of what was added
|
||||
|
||||
## Updating changeNotes in build.gradle.kts
|
||||
|
||||
The `changeNotes` section in `build.gradle.kts` displays on the JetBrains Marketplace plugin page. Follow these rules:
|
||||
|
||||
### Content Requirements
|
||||
- **Match CHANGES.md exactly**: Use the same content from the `[To Be Released]` section
|
||||
- **Don't create a shorter version**: Include all entries as they appear in CHANGES.md
|
||||
- **Keep the same level of detail**: Don't summarize or condense
|
||||
|
||||
### HTML Formatting
|
||||
Convert Markdown to HTML format:
|
||||
- Headers: `### Features:` → `<b>Features:</b>`
|
||||
- Line breaks: Use `<br>` between items
|
||||
- Links: Convert markdown links to HTML `<a href="">` tags
|
||||
- Bullet points: Use `•` or keep `*` with proper spacing
|
||||
- Code blocks: Use `<code>` tags for commands like `<code>gn</code>`
|
||||
|
||||
### Special Notes
|
||||
- **IMPORTANT**: Keep any existing information about the reward program in changeNotes
|
||||
- This content appears in the plugin description on JetBrains Marketplace
|
||||
|
||||
### Example Conversion
|
||||
Markdown in CHANGES.md:
|
||||
```
|
||||
### Features:
|
||||
* Added support for `gn` text object
|
||||
* [VIM-3456](https://youtrack.jetbrains.com/issue/VIM-3456) Fixed cursor position
|
||||
```
|
||||
|
||||
HTML in changeNotes:
|
||||
```html
|
||||
<b>Features:</b><br>
|
||||
• Added support for <code>gn</code> text object<br>
|
||||
• <a href="https://youtrack.jetbrains.com/issue/VIM-3456">VIM-3456</a> Fixed cursor position<br>
|
||||
```
|
||||
|
||||
## Important Notes
|
||||
|
||||
- **Don't create a PR if changelog is already up to date**
|
||||
- **Preserve existing format and structure**
|
||||
- **Maintain chronological order (newest first)**
|
||||
- **Keep the historical gap note between 2.9.0 and 2.28.0**
|
@@ -1,206 +0,0 @@
|
||||
# Codebase Maintenance Instructions
|
||||
|
||||
## Goal
|
||||
|
||||
Perform routine maintenance on random parts of the IdeaVim codebase to ensure code quality, consistency, and catch potential issues early. This is not about being overly pedantic or making changes for the sake of changes - it's about keeping an eye on the codebase and identifying genuine issues.
|
||||
|
||||
## Approach
|
||||
|
||||
### 1. Select Random Area
|
||||
|
||||
Choose a random part of the codebase to inspect. Use one of these strategies:
|
||||
|
||||
```bash
|
||||
# Get a random Kotlin file
|
||||
find . -name "*.kt" -not -path "*/build/*" -not -path "*/.gradle/*" | shuf -n 1
|
||||
|
||||
# Get a random package/directory
|
||||
find . -type d -name "*.kt" -not -path "*/build/*" | shuf -n 1 | xargs dirname
|
||||
|
||||
# Pick from core areas randomly
|
||||
# - vim-engine/src/main/kotlin/com/maddyhome/idea/vim/
|
||||
# - src/main/java/com/maddyhome/idea/vim/
|
||||
# - tests/
|
||||
```
|
||||
|
||||
**Important**: You're not limited to the file you randomly selected. If investigating reveals related files that need attention, follow the trail. The random selection is just a starting point.
|
||||
|
||||
## 2. What to Check
|
||||
|
||||
### Code Style & Formatting
|
||||
- **Kotlin conventions**: Proper use of data classes, sealed classes, when expressions
|
||||
- **Naming consistency**: Follow existing patterns in the codebase
|
||||
- **Import organization**: Remove unused imports, prefer explicit imports over wildcards (wildcard imports are generally not welcome)
|
||||
- **Code structure**: Proper indentation, spacing, line breaks
|
||||
- **Documentation**: KDoc comments where needed (public APIs, complex logic)
|
||||
- **Copyright years**: Do NOT update copyright years unless you're making substantive changes to the file. It's perfectly fine for copyright to show an older year. Don't mention copyright year updates in commit messages or change summaries
|
||||
|
||||
### Code Quality Issues
|
||||
- **Null safety**: Proper use of nullable types, safe calls, Elvis operator
|
||||
- **Error handling**: Appropriate exception handling, meaningful error messages
|
||||
- **Code duplication**: Identify repeated code that could be extracted
|
||||
- **Dead code**: Unused functions, parameters, variables
|
||||
- **TODOs/FIXMEs**: Check if old TODOs are still relevant or can be addressed
|
||||
- **Magic numbers/strings**: Should be named constants
|
||||
- **Complex conditionals**: Can they be simplified or extracted?
|
||||
|
||||
### Potential Bugs
|
||||
- **Off-by-one errors**: Especially in loops and range operations
|
||||
- **Edge cases**: Empty collections, null values, boundary conditions
|
||||
- **Type safety**: Unnecessary casts, unchecked casts
|
||||
- **Resource handling**: Proper cleanup, try-with-resources
|
||||
- **Concurrency issues**: Thread safety if applicable
|
||||
- **State management**: Proper initialization, mutation patterns
|
||||
- **IdeaVim enablement checks**: Verify that `injector.enabler.isEnabled()` or `Editor.isIdeaVimDisabledHere` are not missed in places where they should be checked. These functions determine if IdeaVim is active and should be called before performing Vim-specific operations
|
||||
|
||||
### Architecture & Design
|
||||
- **Separation of concerns**: Does the code have a single responsibility?
|
||||
- **Dependency direction**: Are dependencies pointing the right way?
|
||||
- **Abstraction level**: Consistent level of abstraction within methods
|
||||
- **Vim architecture alignment**: Does it match Vim's design philosophy?
|
||||
- **IntelliJ Platform conventions**: Proper use of platform APIs
|
||||
|
||||
### Testing
|
||||
- **Test coverage**: Are there tests for the code you're reviewing?
|
||||
- If checking a specific command or function, verify that tests exist for it
|
||||
- If tests exist, check if they cover the needed cases (edge cases, error conditions, typical usage)
|
||||
- If tests don't exist or coverage is incomplete, consider creating comprehensive test coverage
|
||||
- **Test quality**: Do tests cover edge cases?
|
||||
- **Test naming**: Clear, descriptive test names
|
||||
- **Flaky tests**: Any potentially unstable tests?
|
||||
- **Regression tests for bug fixes**: When fixing a bug, always write a test that:
|
||||
- Would fail with the old (buggy) implementation
|
||||
- Passes with the fixed implementation
|
||||
- Clearly documents what bug it's testing (include comments explaining the issue)
|
||||
- Tests the specific boundary condition or edge case that exposed the bug
|
||||
- This ensures the bug doesn't resurface in future refactorings
|
||||
|
||||
## 3. Investigation Strategy
|
||||
|
||||
Don't just look at surface-level issues. Dig deeper:
|
||||
|
||||
1. **Read the code**: Understand what it does before suggesting changes
|
||||
2. **Check related files**: Look at callers, implementations, tests
|
||||
3. **Look at git history**: `git log --oneline <file>` to understand context
|
||||
4. **Find related issues**: Search for TODOs, FIXMEs, or commented code
|
||||
5. **Run tests**: If you make changes, ensure tests pass
|
||||
6. **Check YouTrack**: Look for related issues if you find bugs
|
||||
|
||||
## 4. When to Make Changes
|
||||
|
||||
**DO fix**:
|
||||
- Clear bugs or logic errors
|
||||
- Obvious code quality issues (unused imports, etc.)
|
||||
- Misleading or incorrect documentation
|
||||
- Code that violates established patterns
|
||||
- Security vulnerabilities
|
||||
- Performance issues with measurable impact
|
||||
|
||||
**DON'T fix**:
|
||||
- Stylistic preferences if existing code is consistent
|
||||
- Working code just to use "newer" patterns
|
||||
- Minor formatting if it's consistent with surrounding code
|
||||
- Things that are subjective or arguable
|
||||
- Massive refactorings without clear benefit
|
||||
|
||||
**When in doubt**: Document the issue in your report but don't make changes.
|
||||
|
||||
## 5. Making Changes
|
||||
|
||||
If you decide to make changes:
|
||||
|
||||
1. **Make focused commits**: One logical change per commit
|
||||
- If the change affects many files or is complicated, split it into multiple step-by-step commits
|
||||
- This makes it easier for reviewers to understand the changes
|
||||
- Example: First commit renames a function, second commit updates callers, third commit adds new functionality
|
||||
2. **Write clear commit messages**: Explain why, not just what
|
||||
3. **Run tests**: `./gradlew test -x :tests:property-tests:test -x :tests:long-running-tests:test`
|
||||
|
||||
## 6. Examples
|
||||
|
||||
### Good Maintenance Examples
|
||||
|
||||
**Example 1: Found and fixed null safety issue**
|
||||
```
|
||||
Inspected: vim-engine/.../motion/VimMotionHandler.kt
|
||||
|
||||
Issues found:
|
||||
- Several nullable properties accessed without safe checks
|
||||
- Could cause NPE in edge cases with cursor at document end
|
||||
|
||||
Changes:
|
||||
- Added null checks with Elvis operator
|
||||
- Added early returns for invalid state
|
||||
- Added KDoc explaining preconditions
|
||||
```
|
||||
|
||||
**Example 2: No changes needed**
|
||||
```
|
||||
Inspected: src/.../action/change/ChangeLineAction.kt
|
||||
|
||||
Checked:
|
||||
- Code style and formatting ✓
|
||||
- Null safety ✓
|
||||
- Error handling ✓
|
||||
- Tests present and comprehensive ✓
|
||||
|
||||
Observations:
|
||||
- Code is well-structured and follows conventions
|
||||
- Good test coverage including edge cases
|
||||
- Documentation is clear
|
||||
- No issues found
|
||||
```
|
||||
|
||||
**Example 3: Found issues but didn't fix**
|
||||
```
|
||||
Inspected: tests/.../motion/MotionTests.kt
|
||||
|
||||
Issues noted:
|
||||
- Some test names could be more descriptive
|
||||
- Potential for extracting common setup code
|
||||
- Tests are comprehensive but could add edge case for empty file
|
||||
|
||||
Recommendation: These are minor quality-of-life improvements.
|
||||
Not critical, but could be addressed in future cleanup.
|
||||
```
|
||||
|
||||
## IdeaVim-Specific Considerations
|
||||
|
||||
- **Vim compatibility**: Changes should maintain compatibility with Vim behavior
|
||||
- **IntelliJ Platform**: Follow IntelliJ platform conventions and APIs
|
||||
- **Property tests**: Can be flaky - verify if test failures relate to your changes
|
||||
- **Action syntax**: Use `<Action>` in mappings, not `:action`
|
||||
- **Architecture & Guidelines**: Refer to [CONTRIBUTING.md](../CONTRIBUTING.md) for:
|
||||
- Architecture overview and where to find specific code
|
||||
- Testing guidelines and corner cases to consider
|
||||
- Common patterns and conventions
|
||||
- Information about awards for quality contributions
|
||||
|
||||
## Commands Reference
|
||||
|
||||
```bash
|
||||
# Run tests (standard suite)
|
||||
./gradlew test -x :tests:property-tests:test -x :tests:long-running-tests:test
|
||||
|
||||
# Run specific test class
|
||||
./gradlew test --tests "ClassName"
|
||||
|
||||
# Check code style
|
||||
./gradlew ktlintCheck
|
||||
|
||||
# Format code
|
||||
./gradlew ktlintFormat
|
||||
|
||||
# Run IdeaVim in dev instance
|
||||
./gradlew runIde
|
||||
```
|
||||
|
||||
## Final Notes
|
||||
|
||||
- **Be thorough but practical**: Don't waste time on nitpicks
|
||||
- **Context matters**: Understand why code is the way it is before changing
|
||||
- **Quality over quantity**: One good fix is better than ten trivial changes
|
||||
- **Document your process**: Help future maintainers understand your thinking
|
||||
- **Learn from the code**: Use this as an opportunity to understand the codebase better
|
||||
|
||||
Remember: The goal is to keep the codebase healthy, not to achieve perfection. Focus on genuine improvements that make the code safer, clearer, or more maintainable.
|
@@ -1,12 +0,0 @@
|
||||
{
|
||||
"name": "Java",
|
||||
"image": "mcr.microsoft.com/devcontainers/java:1-21",
|
||||
"features": {
|
||||
"ghcr.io/devcontainers/features/java:1": {
|
||||
"version": "none",
|
||||
"installMaven": "true",
|
||||
"mavenVersion": "3.8.6",
|
||||
"installGradle": "true"
|
||||
}
|
||||
}
|
||||
}
|
1
.gitattributes
vendored
Normal file
1
.gitattributes
vendored
Normal file
@@ -0,0 +1 @@
|
||||
* text=auto eol=lf
|
5
.github/workflows/checkNewPlugins.yml
vendored
5
.github/workflows/checkNewPlugins.yml
vendored
@@ -14,16 +14,15 @@ jobs:
|
||||
build:
|
||||
|
||||
runs-on: ubuntu-latest
|
||||
if: github.repository == 'JetBrains/ideavim'
|
||||
|
||||
steps:
|
||||
- name: Fetch origin repo
|
||||
uses: actions/checkout@v3
|
||||
|
||||
- name: Set up JDK 21
|
||||
- name: Set up JDK 17
|
||||
uses: actions/setup-java@v2
|
||||
with:
|
||||
java-version: '21'
|
||||
java-version: '17'
|
||||
distribution: 'adopt'
|
||||
server-id: github # Value of the distributionManagement/repository/id field of the pom.xml
|
||||
settings-path: ${{ github.workspace }} # location for the settings.xml file
|
||||
|
47
.github/workflows/claude-code-review.yml
vendored
47
.github/workflows/claude-code-review.yml
vendored
@@ -1,47 +0,0 @@
|
||||
name: Claude Code Review
|
||||
|
||||
on:
|
||||
pull_request:
|
||||
types: [opened, synchronize]
|
||||
|
||||
jobs:
|
||||
claude-review:
|
||||
runs-on: ubuntu-latest
|
||||
permissions:
|
||||
contents: read
|
||||
pull-requests: write
|
||||
id-token: write
|
||||
|
||||
steps:
|
||||
- name: Checkout repository
|
||||
uses: actions/checkout@v4
|
||||
with:
|
||||
fetch-depth: 1
|
||||
|
||||
- name: Run Claude Code Review
|
||||
id: claude-review
|
||||
uses: anthropics/claude-code-action@v1
|
||||
with:
|
||||
claude_code_oauth_token: ${{ secrets.CLAUDE_CODE_OAUTH_TOKEN }}
|
||||
prompt: |
|
||||
REPO: ${{ github.repository }}
|
||||
PR NUMBER: ${{ github.event.pull_request.number }}
|
||||
CONTRIBUTOR: ${{ github.event.pull_request.user.login }}
|
||||
|
||||
Please review this pull request and provide feedback on:
|
||||
- Code quality and best practices
|
||||
- Potential bugs or issues
|
||||
- Performance considerations
|
||||
- Security concerns
|
||||
- Test coverage
|
||||
|
||||
Provide detailed feedback using inline comments for specific issues.
|
||||
|
||||
Use the repository's CLAUDE.md for guidance on style and conventions. Be constructive and helpful in your feedback.
|
||||
|
||||
Use `gh pr comment` with your Bash tool to leave your review as a comment on the PR.
|
||||
|
||||
# See https://github.com/anthropics/claude-code-action/blob/main/docs/usage.md
|
||||
# or https://docs.anthropic.com/en/docs/claude-code/sdk#command-line for available options
|
||||
claude_args: '--allowed-tools "mcp__github_inline_comment__create_inline_comment,Bash(gh pr comment:*),Bash(gh pr diff:*),Bash(gh pr view:*)"'
|
||||
|
50
.github/workflows/claude.yml
vendored
50
.github/workflows/claude.yml
vendored
@@ -1,50 +0,0 @@
|
||||
name: Claude Code
|
||||
|
||||
on:
|
||||
issue_comment:
|
||||
types: [created]
|
||||
pull_request_review_comment:
|
||||
types: [created]
|
||||
issues:
|
||||
types: [opened, assigned]
|
||||
pull_request_review:
|
||||
types: [submitted]
|
||||
|
||||
jobs:
|
||||
claude:
|
||||
if: |
|
||||
(github.event_name == 'issue_comment' && contains(github.event.comment.body, '@claude')) ||
|
||||
(github.event_name == 'pull_request_review_comment' && contains(github.event.comment.body, '@claude')) ||
|
||||
(github.event_name == 'pull_request_review' && contains(github.event.review.body, '@claude')) ||
|
||||
(github.event_name == 'issues' && (contains(github.event.issue.body, '@claude') || contains(github.event.issue.title, '@claude')))
|
||||
runs-on: ubuntu-latest
|
||||
permissions:
|
||||
contents: read
|
||||
pull-requests: read
|
||||
issues: read
|
||||
id-token: write
|
||||
actions: read # Required for Claude to read CI results on PRs
|
||||
steps:
|
||||
- name: Checkout repository
|
||||
uses: actions/checkout@v4
|
||||
with:
|
||||
fetch-depth: 1
|
||||
|
||||
- name: Run Claude Code
|
||||
id: claude
|
||||
uses: anthropics/claude-code-action@v1
|
||||
with:
|
||||
claude_code_oauth_token: ${{ secrets.CLAUDE_CODE_OAUTH_TOKEN }}
|
||||
|
||||
# This is an optional setting that allows Claude to read CI results on PRs
|
||||
additional_permissions: |
|
||||
actions: read
|
||||
|
||||
# Optional: Give a custom prompt to Claude. If this is not specified, Claude will perform the instructions specified in the comment that tagged it.
|
||||
# prompt: 'Update the pull request description to include a summary of changes.'
|
||||
|
||||
# Optional: Add claude_args to customize behavior and configuration
|
||||
# See https://github.com/anthropics/claude-code-action/blob/main/docs/usage.md
|
||||
# or https://docs.anthropic.com/en/docs/claude-code/sdk#command-line for available options
|
||||
# claude_args: '--model claude-opus-4-1-20250805 --allowed-tools Bash(gh pr:*)'
|
||||
|
10
.github/workflows/closeYoutrackOnCommit.yml
vendored
10
.github/workflows/closeYoutrackOnCommit.yml
vendored
@@ -8,14 +8,10 @@ on:
|
||||
push:
|
||||
branches: [ master ]
|
||||
|
||||
permissions:
|
||||
contents: write
|
||||
|
||||
jobs:
|
||||
build:
|
||||
|
||||
runs-on: ubuntu-latest
|
||||
if: github.repository == 'JetBrains/ideavim'
|
||||
|
||||
steps:
|
||||
- uses: actions/checkout@v3
|
||||
@@ -23,10 +19,10 @@ jobs:
|
||||
fetch-depth: 300
|
||||
- name: Get tags
|
||||
run: git fetch --tags origin
|
||||
- name: Set up JDK 21
|
||||
- name: Set up JDK 11
|
||||
uses: actions/setup-java@v2
|
||||
with:
|
||||
java-version: '21'
|
||||
java-version: '11'
|
||||
distribution: 'adopt'
|
||||
server-id: github # Value of the distributionManagement/repository/id field of the pom.xml
|
||||
settings-path: ${{ github.workspace }} # location for the settings.xml file
|
||||
@@ -37,7 +33,7 @@ jobs:
|
||||
echo "LAST_COMMIT=$(git rev-list -n 1 tags/workflow-close-youtrack)" >> $GITHUB_ENV
|
||||
|
||||
- name: Update YouTrack
|
||||
run: ./gradlew --no-configuration-cache updateYoutrackOnCommit
|
||||
run: ./gradlew updateYoutrackOnCommit
|
||||
env:
|
||||
SUCCESS_COMMIT: ${{ env.LAST_COMMIT }}
|
||||
YOUTRACK_TOKEN: ${{ secrets.YOUTRACK_TOKEN }}
|
||||
|
55
.github/workflows/codebaseMaintenance.yml
vendored
55
.github/workflows/codebaseMaintenance.yml
vendored
@@ -1,55 +0,0 @@
|
||||
name: Codebase Maintenance with Claude
|
||||
|
||||
on:
|
||||
schedule:
|
||||
# Run every day at 6 AM UTC
|
||||
- cron: '0 6 * * *'
|
||||
workflow_dispatch: # Allow manual trigger
|
||||
|
||||
jobs:
|
||||
maintain-codebase:
|
||||
runs-on: ubuntu-latest
|
||||
if: github.repository == 'JetBrains/ideavim'
|
||||
|
||||
permissions:
|
||||
contents: write
|
||||
pull-requests: write
|
||||
id-token: write
|
||||
issues: read
|
||||
actions: read
|
||||
|
||||
steps:
|
||||
- name: Checkout repository
|
||||
uses: actions/checkout@v4
|
||||
with:
|
||||
fetch-depth: 0 # Need history for context
|
||||
|
||||
- name: Run Claude Code for Codebase Maintenance
|
||||
uses: anthropics/claude-code-action@v1
|
||||
with:
|
||||
claude_code_oauth_token: ${{ secrets.CLAUDE_CODE_OAUTH_TOKEN }}
|
||||
|
||||
prompt: |
|
||||
## Task: Perform Codebase Maintenance
|
||||
|
||||
Your goal is to inspect a random part of the IdeaVim codebase and perform maintenance checks.
|
||||
|
||||
Please follow the detailed maintenance instructions in `.claude/maintenance-instructions.md`.
|
||||
|
||||
## Creating Pull Requests
|
||||
|
||||
**Only create a pull request if you made changes to the codebase.**
|
||||
|
||||
If you made changes, create a PR with:
|
||||
- **Title**: "Maintenance: <area> - <brief description>"
|
||||
- Example: "Maintenance: VimMotionHandler - Fix null safety issues"
|
||||
- **Body** including:
|
||||
- What area you inspected
|
||||
- Issues you found
|
||||
- Changes you made
|
||||
- Why the changes improve the code
|
||||
|
||||
If no changes are needed, do not create a pull request.
|
||||
|
||||
# Allow Claude to use necessary tools for code inspection and maintenance
|
||||
claude_args: '--allowed-tools "Read,Edit,Write,Glob,Grep,Bash(git:*),Bash(gh:*),Bash(./gradlew:*),Bash(find:*),Bash(shuf:*)"'
|
19
.github/workflows/codeql-analysis.yml
vendored
19
.github/workflows/codeql-analysis.yml
vendored
@@ -20,11 +20,6 @@ on:
|
||||
schedule:
|
||||
- cron: '44 12 * * 4'
|
||||
|
||||
permissions:
|
||||
actions: read
|
||||
contents: read
|
||||
security-events: write
|
||||
|
||||
jobs:
|
||||
analyze:
|
||||
name: Analyze
|
||||
@@ -39,18 +34,12 @@ jobs:
|
||||
# https://docs.github.com/en/free-pro-team@latest/github/finding-security-vulnerabilities-and-errors-in-your-code/configuring-code-scanning#changing-the-languages-that-are-analyzed
|
||||
|
||||
steps:
|
||||
- name: Setup Java
|
||||
uses: actions/setup-java@v4
|
||||
with:
|
||||
distribution: 'temurin' # See 'Supported distributions' for available options
|
||||
java-version: '21'
|
||||
|
||||
- name: Checkout repository
|
||||
uses: actions/checkout@v4
|
||||
uses: actions/checkout@v2
|
||||
|
||||
# Initializes the CodeQL tools for scanning.
|
||||
- name: Initialize CodeQL
|
||||
uses: github/codeql-action/init@v3
|
||||
uses: github/codeql-action/init@v2
|
||||
with:
|
||||
languages: ${{ matrix.language }}
|
||||
# If you wish to specify custom queries, you can do so here or in a config file.
|
||||
@@ -61,7 +50,7 @@ jobs:
|
||||
# Autobuild attempts to build any compiled languages (C/C++, C#, or Java).
|
||||
# If this step fails, then you should remove it and run the build manually (see below)
|
||||
- name: Autobuild
|
||||
uses: github/codeql-action/autobuild@v3
|
||||
uses: github/codeql-action/autobuild@v2
|
||||
|
||||
# ℹ️ Command-line programs to run using the OS shell.
|
||||
# 📚 https://git.io/JvXDl
|
||||
@@ -75,4 +64,4 @@ jobs:
|
||||
# make release
|
||||
|
||||
- name: Perform CodeQL Analysis
|
||||
uses: github/codeql-action/analyze@v3
|
||||
uses: github/codeql-action/analyze@v2
|
||||
|
7
.github/workflows/integrationsTest.yml
vendored
7
.github/workflows/integrationsTest.yml
vendored
@@ -12,22 +12,21 @@ jobs:
|
||||
build:
|
||||
|
||||
runs-on: ubuntu-latest
|
||||
if: github.repository == 'JetBrains/ideavim'
|
||||
|
||||
steps:
|
||||
- uses: actions/checkout@v2
|
||||
with:
|
||||
fetch-depth: 300
|
||||
- name: Set up JDK 21
|
||||
- name: Set up JDK 11
|
||||
uses: actions/setup-java@v2
|
||||
with:
|
||||
java-version: '21'
|
||||
java-version: '11'
|
||||
distribution: 'adopt'
|
||||
server-id: github # Value of the distributionManagement/repository/id field of the pom.xml
|
||||
settings-path: ${{ github.workspace }} # location for the settings.xml file
|
||||
|
||||
- name: Run tests
|
||||
run: ./gradlew --no-configuration-cache integrationsTest
|
||||
run: ./gradlew integrationsTest
|
||||
env:
|
||||
YOUTRACK_TOKEN: ${{ secrets.YOUTRACK_TOKEN }}
|
||||
GITHUB_OAUTH: ${{ secrets.GITHUB_TOKEN }}
|
||||
|
21
.github/workflows/junie.yml
vendored
21
.github/workflows/junie.yml
vendored
@@ -1,21 +0,0 @@
|
||||
name: Junie
|
||||
run-name: Junie run ${{ inputs.run_id }}
|
||||
|
||||
permissions:
|
||||
contents: write
|
||||
|
||||
on:
|
||||
workflow_dispatch:
|
||||
inputs:
|
||||
run_id:
|
||||
description: "id of workflow process"
|
||||
required: true
|
||||
workflow_params:
|
||||
description: "stringified params"
|
||||
required: true
|
||||
|
||||
jobs:
|
||||
call-workflow-passing-data:
|
||||
uses: jetbrains-junie/junie-workflows/.github/workflows/ej-issue.yml@main
|
||||
with:
|
||||
workflow_params: ${{ inputs.workflow_params }}
|
5
.github/workflows/kover.yml
vendored
5
.github/workflows/kover.yml
vendored
@@ -12,16 +12,15 @@ jobs:
|
||||
build:
|
||||
|
||||
runs-on: ubuntu-latest
|
||||
if: github.repository == 'JetBrains/ideavim'
|
||||
|
||||
steps:
|
||||
- uses: actions/checkout@v2
|
||||
with:
|
||||
fetch-depth: 300
|
||||
- name: Set up JDK 21
|
||||
- name: Set up JDK 11
|
||||
uses: actions/setup-java@v2
|
||||
with:
|
||||
java-version: '21'
|
||||
java-version: '11'
|
||||
distribution: 'adopt'
|
||||
server-id: github # Value of the distributionManagement/repository/id field of the pom.xml
|
||||
settings-path: ${{ github.workspace }} # location for the settings.xml file
|
||||
|
2
.github/workflows/mergeDependabotPR.yml
vendored
2
.github/workflows/mergeDependabotPR.yml
vendored
@@ -8,7 +8,7 @@ permissions:
|
||||
jobs:
|
||||
dependabot:
|
||||
runs-on: ubuntu-latest
|
||||
if: ${{ github.actor == 'dependabot[bot]' && github.repository == 'JetBrains/ideavim' }}
|
||||
if: ${{ github.actor == 'dependabot[bot]' }}
|
||||
steps:
|
||||
- name: Auto-merge Dependabot PR
|
||||
run: gh pr merge --auto --rebase "$PR_URL"
|
||||
|
8
.github/workflows/mergePr.yml
vendored
8
.github/workflows/mergePr.yml
vendored
@@ -11,7 +11,7 @@ on:
|
||||
jobs:
|
||||
build:
|
||||
|
||||
if: false
|
||||
if: github.event.pull_request.merged == true
|
||||
runs-on: ubuntu-latest
|
||||
|
||||
steps:
|
||||
@@ -20,17 +20,17 @@ jobs:
|
||||
fetch-depth: 50
|
||||
# See end of file updateChangeslog.yml for explanation of this secret
|
||||
ssh-key: ${{ secrets.PUSH_TO_PROTECTED_BRANCH_SECRET }}
|
||||
- name: Set up JDK 21
|
||||
- name: Set up JDK 11
|
||||
uses: actions/setup-java@v2
|
||||
with:
|
||||
java-version: '21'
|
||||
java-version: '11'
|
||||
distribution: 'adopt'
|
||||
server-id: github # Value of the distributionManagement/repository/id field of the pom.xml
|
||||
settings-path: ${{ github.workspace }} # location for the settings.xml file
|
||||
|
||||
- name: Update authors
|
||||
id: update_authors
|
||||
run: ./gradlew --no-configuration-cache updateMergedPr -PprId=${{ github.event.number }}
|
||||
run: ./gradlew updateMergedPr -PprId=${{ github.event.number }}
|
||||
env:
|
||||
GITHUB_OAUTH: ${{ secrets.GITHUB_TOKEN }}
|
||||
|
||||
|
81
.github/workflows/runUiOctopusTests.yml
vendored
81
.github/workflows/runUiOctopusTests.yml
vendored
@@ -1,81 +0,0 @@
|
||||
name: Run Non Octopus UI Tests
|
||||
on:
|
||||
workflow_dispatch:
|
||||
schedule:
|
||||
- cron: '0 12 * * *'
|
||||
jobs:
|
||||
build-for-ui-test-mac-os:
|
||||
if: github.repository == 'JetBrains/ideavim'
|
||||
runs-on: macos-latest
|
||||
steps:
|
||||
- uses: actions/checkout@v4
|
||||
- name: Setup Java
|
||||
uses: actions/setup-java@v4
|
||||
with:
|
||||
distribution: zulu
|
||||
java-version: 21
|
||||
- name: Setup FFmpeg
|
||||
run: brew install ffmpeg
|
||||
# - name: Setup Gradle
|
||||
# uses: gradle/gradle-build-action@v2.4.2
|
||||
- name: Build Plugin
|
||||
run: gradle :buildPlugin
|
||||
- name: Run Idea
|
||||
run: |
|
||||
mkdir -p build/reports
|
||||
gradle --no-configuration-cache runIdeForUiTests -Doctopus.handler=false > build/reports/idea.log &
|
||||
- name: Wait for Idea started
|
||||
uses: jtalk/url-health-check-action@v3
|
||||
with:
|
||||
url: http://127.0.0.1:8082
|
||||
max-attempts: 20
|
||||
retry-delay: 10s
|
||||
- name: Tests
|
||||
run: gradle :tests:ui-ij-tests:testUi
|
||||
- name: Move video
|
||||
if: always()
|
||||
run: mv tests/ui-ij-tests/video build/reports
|
||||
- name: Move sandbox logs
|
||||
if: always()
|
||||
run: mv build/idea-sandbox/IC-*/log_runIdeForUiTests idea-sandbox-log
|
||||
- name: Save report
|
||||
if: always()
|
||||
uses: actions/upload-artifact@v4
|
||||
with:
|
||||
name: ui-test-fails-report-mac
|
||||
path: |
|
||||
build/reports
|
||||
tests/ui-ij-tests/build/reports
|
||||
idea-sandbox-log
|
||||
# build-for-ui-test-linux:
|
||||
# runs-on: ubuntu-latest
|
||||
# steps:
|
||||
# - uses: actions/checkout@v2
|
||||
# - name: Setup Java
|
||||
# uses: actions/setup-java@v2.1.0
|
||||
# with:
|
||||
# distribution: zulu
|
||||
# java-version: 11
|
||||
# - name: Build Plugin
|
||||
# run: gradle :buildPlugin
|
||||
# - name: Run Idea
|
||||
# run: |
|
||||
# export DISPLAY=:99.0
|
||||
# Xvfb -ac :99 -screen 0 1920x1080x16 &
|
||||
# mkdir -p build/reports
|
||||
# gradle :runIdeForUiTests #> build/reports/idea.log
|
||||
# - name: Wait for Idea started
|
||||
# uses: jtalk/url-health-check-action@1.5
|
||||
# with:
|
||||
# url: http://127.0.0.1:8082
|
||||
# max-attempts: 15
|
||||
# retry-delay: 30s
|
||||
# - name: Tests
|
||||
# run: gradle :testUi
|
||||
# - name: Save fails report
|
||||
# if: ${{ failure() }}
|
||||
# uses: actions/upload-artifact@v2
|
||||
# with:
|
||||
# name: ui-test-fails-report-linux
|
||||
# path: |
|
||||
# ui-test-example/build/reports
|
52
.github/workflows/runUiPyTests.yml
vendored
52
.github/workflows/runUiPyTests.yml
vendored
@@ -1,52 +0,0 @@
|
||||
name: Run UI PyCharm Tests
|
||||
on:
|
||||
workflow_dispatch:
|
||||
schedule:
|
||||
- cron: '0 12 * * *'
|
||||
jobs:
|
||||
build-for-ui-test-mac-os:
|
||||
if: github.repository == 'JetBrains/ideavim'
|
||||
runs-on: macos-latest
|
||||
steps:
|
||||
- uses: actions/checkout@v4
|
||||
- name: Setup Java
|
||||
uses: actions/setup-java@v4
|
||||
with:
|
||||
distribution: zulu
|
||||
java-version: 21
|
||||
- uses: actions/setup-python@v5
|
||||
with:
|
||||
python-version: '3.10'
|
||||
- name: Setup FFmpeg
|
||||
run: brew install ffmpeg
|
||||
# - name: Setup Gradle
|
||||
# uses: gradle/gradle-build-action@v2.4.2
|
||||
- name: Build Plugin
|
||||
run: gradle :buildPlugin
|
||||
- name: Run Idea
|
||||
run: |
|
||||
mkdir -p build/reports
|
||||
gradle --no-configuration-cache :runIdeForUiTests -PideaType=PC > build/reports/idea.log &
|
||||
- name: Wait for Idea started
|
||||
uses: jtalk/url-health-check-action@v3
|
||||
with:
|
||||
url: http://127.0.0.1:8082
|
||||
max-attempts: 20
|
||||
retry-delay: 10s
|
||||
- name: Tests
|
||||
run: gradle :tests:ui-py-tests:testUi
|
||||
- name: Move video
|
||||
if: always()
|
||||
run: mv tests/ui-py-tests/video build/reports
|
||||
- name: Move sandbox logs
|
||||
if: always()
|
||||
run: mv build/idea-sandbox/PC-*/log_runIdeForUiTests idea-sandbox-log
|
||||
- name: Save report
|
||||
if: always()
|
||||
uses: actions/upload-artifact@v4
|
||||
with:
|
||||
name: ui-test-fails-report-mac
|
||||
path: |
|
||||
build/reports
|
||||
tests/ui-py-tests/build/reports
|
||||
idea-sandbox-log
|
51
.github/workflows/runUiRdTests.yml
vendored
51
.github/workflows/runUiRdTests.yml
vendored
@@ -1,51 +0,0 @@
|
||||
name: Run UI Rider Tests
|
||||
on:
|
||||
workflow_dispatch:
|
||||
schedule:
|
||||
- cron: '0 12 * * *'
|
||||
jobs:
|
||||
build-for-ui-test-mac-os:
|
||||
if: github.repository == 'JetBrains/ideavim'
|
||||
runs-on: macos-latest
|
||||
steps:
|
||||
- uses: actions/checkout@v4
|
||||
- name: Setup Java
|
||||
uses: actions/setup-java@v4
|
||||
with:
|
||||
distribution: zulu
|
||||
java-version: 21
|
||||
- name: Setup FFmpeg
|
||||
run: brew install ffmpeg
|
||||
# - name: Setup Gradle
|
||||
# uses: gradle/gradle-build-action@v2.4.2
|
||||
- name: Build Plugin
|
||||
run: gradle :buildPlugin
|
||||
- name: Run Idea
|
||||
run: |
|
||||
mkdir -p build/reports
|
||||
gradle --no-configuration-cache :runIdeForUiTests -PideaType=RD > build/reports/idea.log &
|
||||
- name: Wait for Idea started
|
||||
uses: jtalk/url-health-check-action@v3
|
||||
with:
|
||||
url: http://127.0.0.1:8082
|
||||
max-attempts: 100
|
||||
retry-delay: 10s
|
||||
- name: Tests
|
||||
run: gradle :tests:ui-rd-tests:testUi
|
||||
env:
|
||||
RIDER_LICENSE: ${{ secrets.RIDER_LICENSE }}
|
||||
- name: Move video
|
||||
if: always()
|
||||
run: mv tests/ui-rd-tests/video build/reports
|
||||
- name: Move sandbox logs
|
||||
if: always()
|
||||
run: mv build/idea-sandbox/RD-*/log_runIdeForUiTests idea-sandbox-log
|
||||
- name: Save report
|
||||
if: always()
|
||||
uses: actions/upload-artifact@v4
|
||||
with:
|
||||
name: ui-test-fails-report-mac
|
||||
path: |
|
||||
build/reports
|
||||
tests/ui-rd-tests/build/reports
|
||||
idea-sandbox-log
|
36
.github/workflows/runUiTests.yml
vendored
36
.github/workflows/runUiTests.yml
vendored
@@ -5,48 +5,44 @@ on:
|
||||
- cron: '0 12 * * *'
|
||||
jobs:
|
||||
build-for-ui-test-mac-os:
|
||||
if: github.repository == 'JetBrains/ideavim'
|
||||
runs-on: macos-latest
|
||||
steps:
|
||||
- uses: actions/checkout@v4
|
||||
- uses: actions/checkout@v2
|
||||
- name: Setup Java
|
||||
uses: actions/setup-java@v4
|
||||
uses: actions/setup-java@v2.1.0
|
||||
with:
|
||||
distribution: zulu
|
||||
java-version: 21
|
||||
java-version: 11
|
||||
- name: Setup FFmpeg
|
||||
run: brew install ffmpeg
|
||||
# - name: Setup Gradle
|
||||
# uses: gradle/gradle-build-action@v2.4.2
|
||||
uses: FedericoCarboni/setup-ffmpeg@v1
|
||||
with:
|
||||
# Not strictly necessary, but it may prevent rate limit
|
||||
# errors especially on GitHub-hosted macos machines.
|
||||
token: ${{ secrets.GITHUB_TOKEN }}
|
||||
- name: Build Plugin
|
||||
run: gradle :buildPlugin
|
||||
- name: Run Idea
|
||||
run: |
|
||||
mkdir -p build/reports
|
||||
gradle --no-configuration-cache runIdeForUiTests > build/reports/idea.log &
|
||||
gradle :runIdeForUiTests > build/reports/idea.log &
|
||||
- name: Wait for Idea started
|
||||
uses: jtalk/url-health-check-action@v3
|
||||
uses: jtalk/url-health-check-action@1.5
|
||||
with:
|
||||
url: http://127.0.0.1:8082
|
||||
max-attempts: 20
|
||||
retry-delay: 10s
|
||||
- name: Tests
|
||||
run: gradle :tests:ui-ij-tests:testUi
|
||||
run: gradle :testUi
|
||||
- name: Move video
|
||||
if: always()
|
||||
run: mv tests/ui-ij-tests/video build/reports
|
||||
- name: Move sandbox logs
|
||||
if: always()
|
||||
run: mv build/idea-sandbox/IC-*/log_runIdeForUiTests idea-sandbox-log
|
||||
- name: Save report
|
||||
if: always()
|
||||
uses: actions/upload-artifact@v4
|
||||
if: ${{ failure() }}
|
||||
run: mv video build/reports
|
||||
- name: Save fails report
|
||||
if: ${{ failure() }}
|
||||
uses: actions/upload-artifact@v2
|
||||
with:
|
||||
name: ui-test-fails-report-mac
|
||||
path: |
|
||||
build/reports
|
||||
tests/ui-ij-tests/build/reports
|
||||
idea-sandbox-log
|
||||
# build-for-ui-test-linux:
|
||||
# runs-on: ubuntu-latest
|
||||
# steps:
|
||||
|
15
.github/workflows/syncDoc.yml
vendored
15
.github/workflows/syncDoc.yml
vendored
@@ -10,14 +10,10 @@ on:
|
||||
push:
|
||||
branches: [ master ]
|
||||
|
||||
permissions:
|
||||
contents: write
|
||||
|
||||
jobs:
|
||||
build:
|
||||
|
||||
runs-on: ubuntu-latest
|
||||
if: github.repository == 'JetBrains/ideavim'
|
||||
|
||||
steps:
|
||||
- name: Fetch origin repo
|
||||
@@ -37,17 +33,6 @@ jobs:
|
||||
id: update_authors
|
||||
run: cp -a origin/doc/. docs
|
||||
|
||||
# The Wiki for github should have no `.md` in references
|
||||
# Otherwise, such links will lead to the raw text.
|
||||
# However, the `.md` should exist to have a navigation in GitHub code.
|
||||
- name: Replace `.md)` with `)`
|
||||
run: |
|
||||
# Define the directory you want to process
|
||||
DIRECTORY="docs"
|
||||
|
||||
# Find all files in the directory and perform the replacement
|
||||
find $DIRECTORY -type f -exec sed -i 's/\.md)/)/g' {} +
|
||||
|
||||
- name: Commit changes
|
||||
uses: stefanzweifel/git-auto-commit-action@v4
|
||||
with:
|
||||
|
33
.github/workflows/updateAffectedRate.yml
vendored
Normal file
33
.github/workflows/updateAffectedRate.yml
vendored
Normal file
@@ -0,0 +1,33 @@
|
||||
# This workflow will build a package using Gradle and then publish it to GitHub packages when a release is created
|
||||
# For more information see: https://github.com/actions/setup-java/blob/main/docs/advanced-usage.md#Publishing-using-gradle
|
||||
|
||||
# This workflow syncs changes from the docs folder of IdeaVim to the IdeaVim.wiki repository
|
||||
|
||||
name: Update Affected Rate field on YouTrack
|
||||
|
||||
on:
|
||||
workflow_dispatch:
|
||||
schedule:
|
||||
- cron: '0 8 * * *'
|
||||
|
||||
jobs:
|
||||
build:
|
||||
|
||||
runs-on: ubuntu-latest
|
||||
|
||||
steps:
|
||||
- name: Fetch origin repo
|
||||
uses: actions/checkout@v3
|
||||
|
||||
- name: Set up JDK 17
|
||||
uses: actions/setup-java@v2
|
||||
with:
|
||||
java-version: '17'
|
||||
distribution: 'adopt'
|
||||
server-id: github # Value of the distributionManagement/repository/id field of the pom.xml
|
||||
settings-path: ${{ github.workspace }} # location for the settings.xml file
|
||||
|
||||
- name: Update YouTrack
|
||||
run: ./gradlew scripts:updateAffectedRates
|
||||
env:
|
||||
YOUTRACK_TOKEN: ${{ secrets.YOUTRACK_TOKEN }}
|
7
.github/workflows/updateAuthors.yml
vendored
7
.github/workflows/updateAuthors.yml
vendored
@@ -15,7 +15,6 @@ jobs:
|
||||
build:
|
||||
|
||||
runs-on: ubuntu-latest
|
||||
if: github.repository == 'JetBrains/ideavim'
|
||||
|
||||
steps:
|
||||
- uses: actions/checkout@v3
|
||||
@@ -25,10 +24,10 @@ jobs:
|
||||
ssh-key: ${{ secrets.PUSH_TO_PROTECTED_BRANCH_SECRET }}
|
||||
- name: Get tags
|
||||
run: git fetch --tags origin
|
||||
- name: Set up JDK 21
|
||||
- name: Set up JDK 11
|
||||
uses: actions/setup-java@v2
|
||||
with:
|
||||
java-version: '21'
|
||||
java-version: '11'
|
||||
distribution: 'adopt'
|
||||
server-id: github # Value of the distributionManagement/repository/id field of the pom.xml
|
||||
settings-path: ${{ github.workspace }} # location for the settings.xml file
|
||||
@@ -40,7 +39,7 @@ jobs:
|
||||
|
||||
- name: Update authors
|
||||
id: update_authors
|
||||
run: ./gradlew --no-configuration-cache updateAuthors --stacktrace
|
||||
run: ./gradlew updateAuthors --stacktrace
|
||||
env:
|
||||
SUCCESS_COMMIT: ${{ env.LAST_COMMIT }}
|
||||
GITHUB_OAUTH: ${{ secrets.GITHUB_TOKEN }}
|
||||
|
10
.github/workflows/updateChangelog.yml
vendored
10
.github/workflows/updateChangelog.yml
vendored
@@ -7,12 +7,14 @@ on:
|
||||
workflow_dispatch:
|
||||
schedule:
|
||||
- cron: '0 10 * * *'
|
||||
# Workflow run on push is disabled to avoid conflicts when merging PR
|
||||
# push:
|
||||
# branches: [ master ]
|
||||
|
||||
jobs:
|
||||
build:
|
||||
|
||||
runs-on: ubuntu-latest
|
||||
if: false
|
||||
|
||||
steps:
|
||||
- uses: actions/checkout@v3
|
||||
@@ -22,10 +24,10 @@ jobs:
|
||||
ssh-key: ${{ secrets.PUSH_TO_PROTECTED_BRANCH_SECRET }}
|
||||
- name: Get tags
|
||||
run: git fetch --tags origin
|
||||
- name: Set up JDK 21
|
||||
- name: Set up JDK 11
|
||||
uses: actions/setup-java@v2
|
||||
with:
|
||||
java-version: '21'
|
||||
java-version: '11'
|
||||
distribution: 'adopt'
|
||||
server-id: github # Value of the distributionManagement/repository/id field of the pom.xml
|
||||
settings-path: ${{ github.workspace }} # location for the settings.xml file
|
||||
@@ -36,7 +38,7 @@ jobs:
|
||||
echo "LAST_COMMIT=$(git rev-list -n 1 tags/workflow-changelog)" >> $GITHUB_ENV
|
||||
|
||||
- name: Update changelog
|
||||
run: ./gradlew --no-configuration-cache updateChangelog
|
||||
run: ./gradlew updateChangelog
|
||||
env:
|
||||
SUCCESS_COMMIT: ${{ env.LAST_COMMIT }}
|
||||
|
||||
|
45
.github/workflows/updateChangelogClaude.yml
vendored
45
.github/workflows/updateChangelogClaude.yml
vendored
@@ -1,45 +0,0 @@
|
||||
name: Update Changelog with Claude
|
||||
|
||||
on:
|
||||
schedule:
|
||||
# Run every day at 5 AM UTC
|
||||
- cron: '0 5 * * *'
|
||||
workflow_dispatch: # Allow manual trigger
|
||||
|
||||
jobs:
|
||||
update-changelog:
|
||||
runs-on: ubuntu-latest
|
||||
if: github.repository == 'JetBrains/ideavim'
|
||||
|
||||
permissions:
|
||||
contents: write
|
||||
pull-requests: write
|
||||
id-token: write
|
||||
issues: read
|
||||
actions: read
|
||||
|
||||
steps:
|
||||
- name: Checkout repository
|
||||
uses: actions/checkout@v4
|
||||
with:
|
||||
fetch-depth: 0 # Need full history to analyze commits and tags
|
||||
|
||||
- name: Run Claude Code to Update Changelog
|
||||
uses: anthropics/claude-code-action@v1
|
||||
with:
|
||||
claude_code_oauth_token: ${{ secrets.CLAUDE_CODE_OAUTH_TOKEN }}
|
||||
|
||||
prompt: |
|
||||
## Task: Update the CHANGES.md Changelog File
|
||||
|
||||
You need to review the latest commits and maintain the changelog file (CHANGES.md) with meaningful changes.
|
||||
|
||||
Please follow the detailed changelog maintenance instructions in `.claude/changelog-instructions.md`.
|
||||
|
||||
If you find changes that need documenting, update CHANGES.md and create a pull request with:
|
||||
- Title: "Update changelog: <super short summary>"
|
||||
Example: "Update changelog: Add gn text object, fix visual mode issues"
|
||||
- Body: Brief summary of what was added
|
||||
|
||||
# Allow Claude to use git, GitHub CLI, and web access for checking releases and tickets
|
||||
claude_args: '--allowed-tools "Read,Edit,Bash(git:*),Bash(gh:*),WebSearch,WebFetch(domain:plugins.jetbrains.com),WebFetch(domain:youtrack.jetbrains.com),WebFetch(domain:github.com)"'
|
5
.github/workflows/updateFormatting.yml
vendored
5
.github/workflows/updateFormatting.yml
vendored
@@ -12,7 +12,6 @@ jobs:
|
||||
build:
|
||||
|
||||
runs-on: ubuntu-latest
|
||||
if: github.repository == 'JetBrains/ideavim'
|
||||
|
||||
steps:
|
||||
- uses: actions/checkout@v2
|
||||
@@ -20,10 +19,10 @@ jobs:
|
||||
fetch-depth: 50
|
||||
# See end of file updateChangeslog.yml for explanation of this secret
|
||||
ssh-key: ${{ secrets.PUSH_TO_PROTECTED_BRANCH_SECRET }}
|
||||
- name: Set up JDK 21
|
||||
- name: Set up JDK 11
|
||||
uses: actions/setup-java@v2
|
||||
with:
|
||||
java-version: '21'
|
||||
java-version: '11'
|
||||
distribution: 'adopt'
|
||||
server-id: github # Value of the distributionManagement/repository/id field of the pom.xml
|
||||
settings-path: ${{ github.workspace }} # location for the settings.xml file
|
||||
|
47
.github/workflows/updateIntellijVersions.yml
vendored
47
.github/workflows/updateIntellijVersions.yml
vendored
@@ -1,47 +0,0 @@
|
||||
name: Update IntelliJ Version Configurations
|
||||
|
||||
on:
|
||||
schedule:
|
||||
# Run three times a year: August 15, April 30, December 1
|
||||
# Times are in UTC
|
||||
- cron: '0 10 15 8 *' # August 15 at 10:00 UTC
|
||||
- cron: '0 10 30 4 *' # April 30 at 10:00 UTC
|
||||
- cron: '0 10 1 12 *' # December 1 at 10:00 UTC
|
||||
workflow_dispatch: # Allow manual trigger for testing
|
||||
|
||||
jobs:
|
||||
update-intellij-versions:
|
||||
runs-on: ubuntu-latest
|
||||
permissions:
|
||||
contents: write
|
||||
pull-requests: write
|
||||
id-token: write
|
||||
|
||||
steps:
|
||||
- name: Checkout repository
|
||||
uses: actions/checkout@v4
|
||||
with:
|
||||
fetch-depth: 1
|
||||
|
||||
- name: Run Claude Code
|
||||
uses: anthropics/claude-code-action@v1
|
||||
with:
|
||||
claude_code_oauth_token: ${{ secrets.CLAUDE_CODE_OAUTH_TOKEN }}
|
||||
|
||||
prompt: |
|
||||
Look at the file `.teamcity/_Self/Project.kt` and check what IntelliJ versions are currently being tested (look for TestingBuildType configurations).
|
||||
|
||||
Based on the current date and existing versions, determine if a new IntelliJ version should be added.
|
||||
IntelliJ releases new versions approximately 3 times per year:
|
||||
- Spring release (x.1) - around March/April
|
||||
- Summer release (x.2) - around July/August
|
||||
- Fall release (x.3) - around November/December
|
||||
|
||||
If a new version should be added:
|
||||
1. Add the new TestingBuildType configuration in chronological order
|
||||
2. Create a pull request with your changes
|
||||
|
||||
The configuration file is located at: `.teamcity/_Self/Project.kt`
|
||||
Look for the section with comment `// Active tests`
|
||||
|
||||
Only add a new version if it doesn't already exist and if it makes sense based on the release schedule.
|
12
.gitignore
vendored
12
.gitignore
vendored
@@ -1,9 +1,7 @@
|
||||
*.swp
|
||||
/.gradle/
|
||||
/.intellijPlatform/
|
||||
|
||||
/.idea/
|
||||
!/.idea/dictionaries/project.xml
|
||||
!/.idea/scopes
|
||||
!/.idea/copyright
|
||||
!/.idea/icon.png
|
||||
@@ -12,9 +10,6 @@
|
||||
!/.idea/runConfigurations
|
||||
!/.idea/codeStyles
|
||||
!/.idea/vcs.xml
|
||||
!/.idea/misc.xml
|
||||
!/.idea/.name
|
||||
!/.idea/gradle.xml
|
||||
|
||||
**/build/
|
||||
**/out/
|
||||
@@ -27,12 +22,7 @@
|
||||
.teamcity/*.iml
|
||||
|
||||
# Generated by gradle task "generateGrammarSource"
|
||||
vim-engine/src/main/java/com/maddyhome/idea/vim/parser/generated
|
||||
vim-engine/src/main/java/com/maddyhome/idea/vim/regexp/parser/generated
|
||||
src/main/java/com/maddyhome/idea/vim/vimscript/parser/generated
|
||||
|
||||
# Created by github automation
|
||||
settings.xml
|
||||
|
||||
.kotlin
|
||||
|
||||
.claude/settings.local.json
|
||||
|
1
.idea/.name
generated
1
.idea/.name
generated
@@ -1 +0,0 @@
|
||||
IdeaVim
|
2
.idea/copyright/IdeaVim.xml
generated
2
.idea/copyright/IdeaVim.xml
generated
@@ -1,6 +1,6 @@
|
||||
<component name="CopyrightManager">
|
||||
<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" />
|
||||
</copyright>
|
||||
</component>
|
7
.idea/dictionaries/project.xml
generated
7
.idea/dictionaries/project.xml
generated
@@ -1,7 +0,0 @@
|
||||
<component name="ProjectDictionaryState">
|
||||
<dictionary name="project">
|
||||
<words>
|
||||
<w>overstrike</w>
|
||||
</words>
|
||||
</dictionary>
|
||||
</component>
|
28
.idea/gradle.xml
generated
28
.idea/gradle.xml
generated
@@ -1,28 +0,0 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<project version="4">
|
||||
<component name="GradleMigrationSettings" migrationVersion="1" />
|
||||
<component name="GradleSettings">
|
||||
<option name="linkedExternalProjectsSettings">
|
||||
<GradleProjectSettings>
|
||||
<option name="externalProjectPath" value="$PROJECT_DIR$" />
|
||||
<option name="modules">
|
||||
<set>
|
||||
<option value="$PROJECT_DIR$" />
|
||||
<option value="$PROJECT_DIR$/annotation-processors" />
|
||||
<option value="$PROJECT_DIR$/api" />
|
||||
<option value="$PROJECT_DIR$/scripts" />
|
||||
<option value="$PROJECT_DIR$/tests" />
|
||||
<option value="$PROJECT_DIR$/tests/java-tests" />
|
||||
<option value="$PROJECT_DIR$/tests/long-running-tests" />
|
||||
<option value="$PROJECT_DIR$/tests/property-tests" />
|
||||
<option value="$PROJECT_DIR$/tests/ui-fixtures" />
|
||||
<option value="$PROJECT_DIR$/tests/ui-ij-tests" />
|
||||
<option value="$PROJECT_DIR$/tests/ui-py-tests" />
|
||||
<option value="$PROJECT_DIR$/tests/ui-rd-tests" />
|
||||
<option value="$PROJECT_DIR$/vim-engine" />
|
||||
</set>
|
||||
</option>
|
||||
</GradleProjectSettings>
|
||||
</option>
|
||||
</component>
|
||||
</project>
|
22
.idea/misc.xml
generated
22
.idea/misc.xml
generated
@@ -1,22 +0,0 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<project version="4">
|
||||
<component name="EntryPointsManager">
|
||||
<list size="3">
|
||||
<item index="0" class="java.lang.String" itemvalue="com.intellij.vim.annotations.CommandOrMotion" />
|
||||
<item index="1" class="java.lang.String" itemvalue="com.intellij.vim.annotations.ExCommand" />
|
||||
<item index="2" class="java.lang.String" itemvalue="com.intellij.vim.annotations.VimscriptFunction" />
|
||||
</list>
|
||||
</component>
|
||||
<component name="ExternalStorageConfigurationManager" enabled="true" />
|
||||
<component name="FrameworkDetectionExcludesConfiguration">
|
||||
<file type="web" url="file://$PROJECT_DIR$" />
|
||||
</component>
|
||||
<component name="MavenProjectsManager">
|
||||
<option name="originalFiles">
|
||||
<list>
|
||||
<option value="$PROJECT_DIR$/.teamcity/pom.xml" />
|
||||
</list>
|
||||
</option>
|
||||
</component>
|
||||
<component name="ProjectRootManager" version="2" languageLevel="JDK_21" default="true" project-jdk-name="corretto-21" project-jdk-type="JavaSDK" />
|
||||
</project>
|
@@ -12,7 +12,7 @@
|
||||
<option name="taskNames">
|
||||
<list>
|
||||
<option value="check" />
|
||||
<option value="verifyPlugin" />
|
||||
<option value="runPluginVerifier" />
|
||||
</list>
|
||||
</option>
|
||||
<option name="vmOptions" value="" />
|
||||
@@ -20,7 +20,6 @@
|
||||
<ExternalSystemDebugServerProcess>true</ExternalSystemDebugServerProcess>
|
||||
<ExternalSystemReattachDebugProcess>true</ExternalSystemReattachDebugProcess>
|
||||
<DebugAllEnabled>false</DebugAllEnabled>
|
||||
<RunAsTest>false</RunAsTest>
|
||||
<method v="2" />
|
||||
</configuration>
|
||||
</component>
|
3
.idea/runConfigurations/IdeaVim_tests.xml
generated
3
.idea/runConfigurations/IdeaVim_tests.xml
generated
@@ -5,7 +5,7 @@
|
||||
<option name="executionName" />
|
||||
<option name="externalProjectPath" value="$PROJECT_DIR$" />
|
||||
<option name="externalSystemIdString" value="GRADLE" />
|
||||
<option name="scriptParameters" value="-x :tests:property-tests:test -x :tests:long-running-tests:test" />
|
||||
<option name="scriptParameters" value="" />
|
||||
<option name="taskDescriptions">
|
||||
<list />
|
||||
</option>
|
||||
@@ -19,7 +19,6 @@
|
||||
<ExternalSystemDebugServerProcess>true</ExternalSystemDebugServerProcess>
|
||||
<ExternalSystemReattachDebugProcess>true</ExternalSystemReattachDebugProcess>
|
||||
<DebugAllEnabled>false</DebugAllEnabled>
|
||||
<RunAsTest>false</RunAsTest>
|
||||
<method v="2" />
|
||||
</configuration>
|
||||
</component>
|
@@ -1,25 +0,0 @@
|
||||
<component name="ProjectRunConfigurationManager">
|
||||
<configuration default="false" name="Start IJ with IdeaVim (Split Mode)" type="GradleRunConfiguration" factoryName="Gradle">
|
||||
<log_file alias="idea.log" path="$PROJECT_DIR$/build/idea-sandbox/system/log/idea.log" />
|
||||
<ExternalSystemSettings>
|
||||
<option name="executionName" />
|
||||
<option name="externalProjectPath" value="$PROJECT_DIR$" />
|
||||
<option name="externalSystemIdString" value="GRADLE" />
|
||||
<option name="scriptParameters" value="" />
|
||||
<option name="taskDescriptions">
|
||||
<list />
|
||||
</option>
|
||||
<option name="taskNames">
|
||||
<list>
|
||||
<option value="runIdeSplitMode" />
|
||||
</list>
|
||||
</option>
|
||||
<option name="vmOptions" value="" />
|
||||
</ExternalSystemSettings>
|
||||
<ExternalSystemDebugServerProcess>true</ExternalSystemDebugServerProcess>
|
||||
<ExternalSystemReattachDebugProcess>true</ExternalSystemReattachDebugProcess>
|
||||
<DebugAllEnabled>false</DebugAllEnabled>
|
||||
<RunAsTest>false</RunAsTest>
|
||||
<method v="2" />
|
||||
</configuration>
|
||||
</component>
|
2
.idea/vcs.xml
generated
2
.idea/vcs.xml
generated
@@ -11,6 +11,6 @@
|
||||
</option>
|
||||
</component>
|
||||
<component name="VcsDirectoryMappings">
|
||||
<mapping directory="" vcs="Git" />
|
||||
<mapping directory="$PROJECT_DIR$" vcs="Git" />
|
||||
</component>
|
||||
</project>
|
16
.teamcity/_Self/Constants.kt
vendored
16
.teamcity/_Self/Constants.kt
vendored
@@ -5,11 +5,15 @@ object Constants {
|
||||
const val EAP_CHANNEL = "eap"
|
||||
const val DEV_CHANNEL = "Dev"
|
||||
|
||||
const val NVIM_TESTS = "2025.1"
|
||||
const val PROPERTY_TESTS = "2025.1"
|
||||
const val LONG_RUNNING_TESTS = "2025.1"
|
||||
const val RELEASE = "2025.1"
|
||||
const val VERSION = "2.4.0"
|
||||
|
||||
const val RELEASE_DEV = "2025.1"
|
||||
const val RELEASE_EAP = "2025.1"
|
||||
const val GITHUB_TESTS = "LATEST-EAP-SNAPSHOT"
|
||||
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"
|
||||
}
|
||||
|
27
.teamcity/_Self/Project.kt
vendored
27
.teamcity/_Self/Project.kt
vendored
@@ -5,57 +5,46 @@ import _Self.buildTypes.LongRunning
|
||||
import _Self.buildTypes.Nvim
|
||||
import _Self.buildTypes.PluginVerifier
|
||||
import _Self.buildTypes.PropertyBased
|
||||
import _Self.buildTypes.RandomOrderTests
|
||||
import _Self.buildTypes.Qodana
|
||||
import _Self.buildTypes.TestingBuildType
|
||||
import _Self.subprojects.GitHub
|
||||
import _Self.subprojects.OldTests
|
||||
import _Self.subprojects.Releases
|
||||
import _Self.vcsRoots.GitHubPullRequest
|
||||
import _Self.vcsRoots.ReleasesVcsRoot
|
||||
import jetbrains.buildServer.configs.kotlin.v2019_2.BuildType
|
||||
import jetbrains.buildServer.configs.kotlin.v2019_2.Project
|
||||
|
||||
object Project : Project({
|
||||
description = "Vim engine for JetBrains IDEs"
|
||||
description = "Vim engine for IDEs based on the IntelliJ platform"
|
||||
|
||||
subProject(Releases)
|
||||
subProject(GitHub)
|
||||
subProjects(Releases, OldTests, GitHub)
|
||||
|
||||
// VCS roots
|
||||
vcsRoot(GitHubPullRequest)
|
||||
vcsRoot(ReleasesVcsRoot)
|
||||
|
||||
// Active tests
|
||||
buildType(TestingBuildType("Latest EAP", version = "LATEST-EAP-SNAPSHOT"))
|
||||
buildType(TestingBuildType("2025.1"))
|
||||
buildType(TestingBuildType("2025.2"))
|
||||
buildType(TestingBuildType("Latest EAP", "<default>", version = "LATEST-EAP-SNAPSHOT"))
|
||||
buildType(TestingBuildType("Latest EAP With Xorg", "<default>", version = "LATEST-EAP-SNAPSHOT"))
|
||||
|
||||
buildType(PropertyBased)
|
||||
buildType(LongRunning)
|
||||
buildType(RandomOrderTests)
|
||||
|
||||
buildType(Nvim)
|
||||
buildType(PluginVerifier)
|
||||
buildType(Compatibility)
|
||||
|
||||
buildType(Qodana)
|
||||
})
|
||||
|
||||
// Common build type for all configurations
|
||||
abstract class IdeaVimBuildType(init: BuildType.() -> Unit) : BuildType({
|
||||
artifactRules = """
|
||||
+:build/reports => build/reports
|
||||
+:tests/java-tests/build/reports => java-tests/build/reports
|
||||
+:tests/long-running-tests/build/reports => long-running-tests/build/reports
|
||||
+:tests/property-tests/build/reports => property-tests/build/reports
|
||||
+:/mnt/agent/temp/buildTmp/ => /mnt/agent/temp/buildTmp/
|
||||
""".trimIndent()
|
||||
|
||||
init()
|
||||
|
||||
requirements {
|
||||
// These requirements define Linux-Medium configuration.
|
||||
// Unfortunately, requirement by name (teamcity.agent.name) doesn't work
|
||||
// IDK the reason for it, but on our agents this property is empty
|
||||
equals("teamcity.agent.hardware.cpuCount", "16")
|
||||
equals("teamcity.agent.hardware.cpuCount", "4")
|
||||
equals("teamcity.agent.os.family", "Linux")
|
||||
}
|
||||
|
||||
|
24
.teamcity/_Self/buildTypes/Compatibility.kt
vendored
24
.teamcity/_Self/buildTypes/Compatibility.kt
vendored
@@ -20,7 +20,7 @@ object Compatibility : IdeaVimBuildType({
|
||||
name = "Load Verifier"
|
||||
scriptContent = """
|
||||
mkdir verifier1
|
||||
curl -f -L -o verifier1/verifier-cli-dev-all-2.jar "https://packages.jetbrains.team/files/p/ideavim/plugin-verifier/verifier-cli-dev-all-2.jar"
|
||||
curl -f -L -o verifier1/verifier-cli-dev-all-1.jar "https://packages.jetbrains.team/files/p/ideavim/plugin-verifier/verifier-cli-dev-all-1.jar"
|
||||
""".trimIndent()
|
||||
}
|
||||
script {
|
||||
@@ -33,22 +33,14 @@ object Compatibility : IdeaVimBuildType({
|
||||
# Upload verifier-cli-dev-all.jar artifact to the repo in IdeaVim space repo
|
||||
|
||||
java --version
|
||||
java -jar verifier1/verifier-cli-dev-all-2.jar check-plugin '${'$'}org.jetbrains.IdeaVim-EasyMotion' [latest-IU] -team-city
|
||||
java -jar verifier1/verifier-cli-dev-all-2.jar check-plugin '${'$'}eu.theblob42.idea.whichkey' [latest-IU] -team-city
|
||||
java -jar verifier1/verifier-cli-dev-all-2.jar check-plugin '${'$'}IdeaVimExtension' [latest-IU] -team-city
|
||||
java -jar verifier1/verifier-cli-dev-all-1.jar check-plugin '${'$'}org.jetbrains.IdeaVim-EasyMotion' [latest-IU] -team-city
|
||||
java -jar verifier1/verifier-cli-dev-all-1.jar check-plugin '${'$'}io.github.mishkun.ideavimsneak' [latest-IU] -team-city
|
||||
java -jar verifier1/verifier-cli-dev-all-1.jar check-plugin '${'$'}eu.theblob42.idea.whichkey' [latest-IU] -team-city
|
||||
java -jar verifier1/verifier-cli-dev-all-1.jar check-plugin '${'$'}IdeaVimExtension' [latest-IU] -team-city
|
||||
# Outdated java -jar verifier/verifier-cli-dev-all.jar check-plugin '${'$'}github.zgqq.intellij-enhance' [latest-IU] -team-city
|
||||
# java -jar verifier1/verifier-cli-dev-all-2.jar check-plugin '${'$'}com.github.copilot' [latest-IU] -team-city
|
||||
java -jar verifier1/verifier-cli-dev-all-2.jar check-plugin '${'$'}com.github.dankinsoid.multicursor' [latest-IU] -team-city
|
||||
java -jar verifier1/verifier-cli-dev-all-2.jar check-plugin '${'$'}com.joshestein.ideavim-quickscope' [latest-IU] -team-city
|
||||
java -jar verifier1/verifier-cli-dev-all-2.jar check-plugin '${'$'}com.julienphalip.ideavim.peekaboo' [latest-IU] -team-city
|
||||
java -jar verifier1/verifier-cli-dev-all-2.jar check-plugin '${'$'}com.julienphalip.ideavim.switch' [latest-IU] -team-city
|
||||
java -jar verifier1/verifier-cli-dev-all-2.jar check-plugin '${'$'}com.julienphalip.ideavim.functiontextobj' [latest-IU] -team-city
|
||||
java -jar verifier1/verifier-cli-dev-all-2.jar check-plugin '${'$'}com.miksuki.HighlightCursor' [latest-IU] -team-city
|
||||
java -jar verifier1/verifier-cli-dev-all-2.jar check-plugin '${'$'}com.ugarosa.idea.edgemotion' [latest-IU] -team-city
|
||||
java -jar verifier1/verifier-cli-dev-all-2.jar check-plugin '${'$'}cn.mumukehao.plugin' [latest-IU] -team-city
|
||||
java -jar verifier1/verifier-cli-dev-all-2.jar check-plugin '${'$'}com.magidc.ideavim.dial' [latest-IU] -team-city
|
||||
java -jar verifier1/verifier-cli-dev-all-2.jar check-plugin '${'$'}dev.ghostflyby.ideavim.toggleIME' [latest-IU] -team-city
|
||||
java -jar verifier1/verifier-cli-dev-all-2.jar check-plugin '${'$'}com.magidc.ideavim.anyObject' [latest-IU] -team-city
|
||||
java -jar verifier1/verifier-cli-dev-all-1.jar check-plugin '${'$'}com.github.copilot' [latest-IU] -team-city
|
||||
java -jar verifier1/verifier-cli-dev-all-1.jar check-plugin '${'$'}com.github.dankinsoid.multicursor' [latest-IU] -team-city
|
||||
java -jar verifier1/verifier-cli-dev-all-1.jar check-plugin '${'$'}com.joshestein.ideavim-quickscope' [latest-IU] -team-city
|
||||
""".trimIndent()
|
||||
}
|
||||
}
|
||||
|
3
.teamcity/_Self/buildTypes/LongRunning.kt
vendored
3
.teamcity/_Self/buildTypes/LongRunning.kt
vendored
@@ -25,10 +25,9 @@ object LongRunning : IdeaVimBuildType({
|
||||
|
||||
steps {
|
||||
gradle {
|
||||
tasks = "clean :tests:long-running-tests:test"
|
||||
tasks = "clean testLongRunning"
|
||||
buildFile = ""
|
||||
enableStacktrace = true
|
||||
jdkHome = "/usr/lib/jvm/java-21-amazon-corretto"
|
||||
}
|
||||
}
|
||||
|
||||
|
11
.teamcity/_Self/buildTypes/Nvim.kt
vendored
11
.teamcity/_Self/buildTypes/Nvim.kt
vendored
@@ -18,7 +18,7 @@ object Nvim : IdeaVimBuildType({
|
||||
param("env.ORG_GRADLE_PROJECT_downloadIdeaSources", "false")
|
||||
param("env.ORG_GRADLE_PROJECT_ideaVersion", NVIM_TESTS)
|
||||
param("env.ORG_GRADLE_PROJECT_instrumentPluginCode", "false")
|
||||
param("env.ideavim.nvim.path", "./nvim-linux-x86_64/bin/nvim")
|
||||
param("env.ideavim.nvim.path", "./nvim-linux64/bin/nvim")
|
||||
}
|
||||
|
||||
vcs {
|
||||
@@ -32,17 +32,16 @@ object Nvim : IdeaVimBuildType({
|
||||
script {
|
||||
name = "Set up NeoVim"
|
||||
scriptContent = """
|
||||
wget https://github.com/neovim/neovim/releases/download/nightly/nvim-linux-x86_64.tar.gz
|
||||
tar xzf nvim-linux-x86_64.tar.gz
|
||||
cd nvim-linux-x86_64/bin
|
||||
wget https://github.com/neovim/neovim/releases/download/v0.7.2/nvim-linux64.tar.gz
|
||||
tar xzf nvim-linux64.tar.gz
|
||||
cd nvim-linux64/bin
|
||||
chmod +x nvim
|
||||
""".trimIndent()
|
||||
}
|
||||
gradle {
|
||||
tasks = "clean test -x :tests:property-tests:test -x :tests:long-running-tests:test -Dnvim"
|
||||
tasks = "clean testWithNeovim"
|
||||
buildFile = ""
|
||||
enableStacktrace = true
|
||||
jdkHome = "/usr/lib/jvm/java-21-amazon-corretto"
|
||||
}
|
||||
}
|
||||
|
||||
|
3
.teamcity/_Self/buildTypes/PluginVerifier.kt
vendored
3
.teamcity/_Self/buildTypes/PluginVerifier.kt
vendored
@@ -22,10 +22,9 @@ object PluginVerifier : IdeaVimBuildType({
|
||||
|
||||
steps {
|
||||
gradle {
|
||||
tasks = "clean verifyPlugin"
|
||||
tasks = "clean runPluginVerifier"
|
||||
buildFile = ""
|
||||
enableStacktrace = true
|
||||
jdkHome = "/usr/lib/jvm/java-21-amazon-corretto"
|
||||
}
|
||||
}
|
||||
|
||||
|
4
.teamcity/_Self/buildTypes/PropertyBased.kt
vendored
4
.teamcity/_Self/buildTypes/PropertyBased.kt
vendored
@@ -24,11 +24,9 @@ object PropertyBased : IdeaVimBuildType({
|
||||
|
||||
steps {
|
||||
gradle {
|
||||
clearConditions()
|
||||
tasks = "clean :tests:property-tests:test"
|
||||
tasks = "clean testPropertyBased"
|
||||
buildFile = ""
|
||||
enableStacktrace = true
|
||||
jdkHome = "/usr/lib/jvm/java-21-amazon-corretto"
|
||||
}
|
||||
}
|
||||
|
||||
|
@@ -20,13 +20,13 @@ object PublishVimEngine : IdeaVimBuildType({
|
||||
params {
|
||||
param("env.ORG_GRADLE_PROJECT_engineVersion", "%build.number%")
|
||||
param("env.ORG_GRADLE_PROJECT_uploadUrl", "https://packages.jetbrains.team/maven/p/ij/intellij-dependencies")
|
||||
password("env.ORG_GRADLE_PROJECT_spacePassword", "credentialsJSON:5ea56f8c-efe7-4e1e-83de-0c02bcc39d0b", display = ParameterDisplay.HIDDEN)
|
||||
param("env.ORG_GRADLE_PROJECT_spaceUsername", "a121c67e-39ac-40e6-bf82-649855ec27b6")
|
||||
password("env.ORG_GRADLE_PROJECT_spacePassword", "credentialsJSON:790b4e43-ee83-4184-b81b-678afab60409", display = ParameterDisplay.HIDDEN)
|
||||
param("env.ORG_GRADLE_PROJECT_spaceUsername", "Aleksei.Plate")
|
||||
}
|
||||
|
||||
vcs {
|
||||
root(DslContext.settingsRoot)
|
||||
branchFilter = "+:fleet"
|
||||
branchFilter = "+:<default>"
|
||||
|
||||
checkoutMode = CheckoutMode.AUTO
|
||||
}
|
||||
@@ -36,7 +36,6 @@ object PublishVimEngine : IdeaVimBuildType({
|
||||
tasks = ":vim-engine:publish"
|
||||
buildFile = ""
|
||||
enableStacktrace = true
|
||||
jdkHome = "/usr/lib/jvm/java-21-amazon-corretto"
|
||||
}
|
||||
}
|
||||
|
||||
|
81
.teamcity/_Self/buildTypes/Qodana.kt
vendored
Normal file
81
.teamcity/_Self/buildTypes/Qodana.kt
vendored
Normal file
@@ -0,0 +1,81 @@
|
||||
package _Self.buildTypes
|
||||
|
||||
import _Self.Constants.QODANA_TESTS
|
||||
import _Self.IdeaVimBuildType
|
||||
import jetbrains.buildServer.configs.kotlin.v2019_2.CheckoutMode
|
||||
import jetbrains.buildServer.configs.kotlin.v2019_2.DslContext
|
||||
import jetbrains.buildServer.configs.kotlin.v2019_2.buildSteps.Qodana
|
||||
import jetbrains.buildServer.configs.kotlin.v2019_2.buildSteps.gradle
|
||||
import jetbrains.buildServer.configs.kotlin.v2019_2.buildSteps.qodana
|
||||
import jetbrains.buildServer.configs.kotlin.v2019_2.failureConditions.BuildFailureOnMetric
|
||||
import jetbrains.buildServer.configs.kotlin.v2019_2.failureConditions.failOnMetricChange
|
||||
import jetbrains.buildServer.configs.kotlin.v2019_2.triggers.schedule
|
||||
import jetbrains.buildServer.configs.kotlin.v2019_2.triggers.vcs
|
||||
|
||||
object Qodana : IdeaVimBuildType({
|
||||
name = "Qodana checks"
|
||||
params {
|
||||
param("env.ORG_GRADLE_PROJECT_downloadIdeaSources", "false")
|
||||
param("env.ORG_GRADLE_PROJECT_ideaVersion", QODANA_TESTS)
|
||||
param("env.ORG_GRADLE_PROJECT_instrumentPluginCode", "false")
|
||||
}
|
||||
|
||||
vcs {
|
||||
root(DslContext.settingsRoot)
|
||||
branchFilter = "+:<default>"
|
||||
|
||||
checkoutMode = CheckoutMode.AUTO
|
||||
}
|
||||
|
||||
steps {
|
||||
gradle {
|
||||
name = "Generate grammar"
|
||||
tasks = "generateGrammarSource"
|
||||
}
|
||||
qodana {
|
||||
name = "Qodana"
|
||||
param("clonefinder-languages", "")
|
||||
param("collect-anonymous-statistics", "")
|
||||
param("licenseaudit-enable", "")
|
||||
param("clonefinder-languages-container", "")
|
||||
param("linterVersion", "")
|
||||
param("clonefinder-queried-project", "")
|
||||
param("clonefinder-enable", "")
|
||||
param("clonefinder-reference-projects", "")
|
||||
linter = jvm {
|
||||
version = Qodana.JVMVersion.LATEST
|
||||
}
|
||||
reportAsTests = true
|
||||
additionalDockerArguments = "-e QODANA_TOKEN=eyJ0eXAiOiJKV1QiLCJhbGciOiJIUzI1NiJ9.eyJvcmdhbml6YXRpb24iOiIzUFZrQSIsInByb2plY3QiOiIzN1FlQSIsInRva2VuIjoiM0t2bXoifQ.uohp81tM7iAfvvB6k8faarfpV-OjusAaEbWQ8iNrOgs"
|
||||
additionalQodanaArguments = "--baseline qodana.sarif.json"
|
||||
}
|
||||
}
|
||||
|
||||
triggers {
|
||||
vcs {
|
||||
enabled = false
|
||||
branchFilter = "+:<default>"
|
||||
}
|
||||
schedule {
|
||||
schedulingPolicy = daily {
|
||||
hour = 12
|
||||
minute = 0
|
||||
timezone = "SERVER"
|
||||
}
|
||||
param("dayOfWeek", "Sunday")
|
||||
enabled = false
|
||||
}
|
||||
}
|
||||
|
||||
failureConditions {
|
||||
failOnMetricChange {
|
||||
threshold = 0
|
||||
units = BuildFailureOnMetric.MetricUnit.DEFAULT_UNIT
|
||||
comparison = BuildFailureOnMetric.MetricComparison.MORE
|
||||
compareTo = value()
|
||||
metric = BuildFailureOnMetric.MetricType.TEST_FAILED_COUNT
|
||||
param("metricKey", "QodanaProblemsNew")
|
||||
enabled = false
|
||||
}
|
||||
}
|
||||
})
|
44
.teamcity/_Self/buildTypes/RandomOrderTests.kt
vendored
44
.teamcity/_Self/buildTypes/RandomOrderTests.kt
vendored
@@ -1,44 +0,0 @@
|
||||
package _Self.buildTypes
|
||||
|
||||
import _Self.IdeaVimBuildType
|
||||
import jetbrains.buildServer.configs.kotlin.v2019_2.CheckoutMode
|
||||
import jetbrains.buildServer.configs.kotlin.v2019_2.DslContext
|
||||
import jetbrains.buildServer.configs.kotlin.v2019_2.buildSteps.gradle
|
||||
import jetbrains.buildServer.configs.kotlin.v2019_2.triggers.vcs
|
||||
|
||||
object RandomOrderTests : IdeaVimBuildType({
|
||||
name = "Random order tests"
|
||||
description = "Running tests with random order on each run. This way we can catch order-dependent bugs."
|
||||
params {
|
||||
param("env.ORG_GRADLE_PROJECT_downloadIdeaSources", "false")
|
||||
param("env.ORG_GRADLE_PROJECT_instrumentPluginCode", "false")
|
||||
}
|
||||
|
||||
vcs {
|
||||
root(DslContext.settingsRoot)
|
||||
branchFilter = "+:<default>"
|
||||
|
||||
checkoutMode = CheckoutMode.AUTO
|
||||
}
|
||||
|
||||
steps {
|
||||
gradle {
|
||||
tasks = """
|
||||
clean test
|
||||
-x :tests:property-tests:test
|
||||
-x :tests:long-running-tests:test
|
||||
-Djunit.jupiter.execution.order.random.seed=default
|
||||
-Djunit.jupiter.testmethod.order.default=random
|
||||
""".trimIndent().replace("\n", " ")
|
||||
buildFile = ""
|
||||
enableStacktrace = true
|
||||
jdkHome = "/usr/lib/jvm/java-21-amazon-corretto"
|
||||
}
|
||||
}
|
||||
|
||||
triggers {
|
||||
vcs {
|
||||
branchFilter = "+:<default>"
|
||||
}
|
||||
}
|
||||
})
|
3
.teamcity/_Self/buildTypes/ReleaseDev.kt
vendored
3
.teamcity/_Self/buildTypes/ReleaseDev.kt
vendored
@@ -47,16 +47,13 @@ object ReleaseDev : IdeaVimBuildType({
|
||||
gradle {
|
||||
name = "Calculate new dev version"
|
||||
tasks = "scripts:calculateNewDevVersion"
|
||||
jdkHome = "/usr/lib/jvm/java-21-amazon-corretto"
|
||||
}
|
||||
gradle {
|
||||
name = "Set TeamCity build number"
|
||||
tasks = "scripts:setTeamCityBuildNumber"
|
||||
jdkHome = "/usr/lib/jvm/java-21-amazon-corretto"
|
||||
}
|
||||
gradle {
|
||||
tasks = "publishPlugin"
|
||||
jdkHome = "/usr/lib/jvm/java-21-amazon-corretto"
|
||||
}
|
||||
}
|
||||
|
||||
|
21
.teamcity/_Self/buildTypes/ReleaseEap.kt
vendored
21
.teamcity/_Self/buildTypes/ReleaseEap.kt
vendored
@@ -5,7 +5,6 @@ import _Self.Constants.RELEASE_EAP
|
||||
import _Self.IdeaVimBuildType
|
||||
import jetbrains.buildServer.configs.kotlin.v2019_2.CheckoutMode
|
||||
import jetbrains.buildServer.configs.kotlin.v2019_2.DslContext
|
||||
import jetbrains.buildServer.configs.kotlin.v2019_2.ParameterDisplay
|
||||
import jetbrains.buildServer.configs.kotlin.v2019_2.buildFeatures.sshAgent
|
||||
import jetbrains.buildServer.configs.kotlin.v2019_2.buildSteps.gradle
|
||||
import jetbrains.buildServer.configs.kotlin.v2019_2.buildSteps.script
|
||||
@@ -31,11 +30,6 @@ object ReleaseEap : IdeaVimBuildType({
|
||||
"credentialsJSON:a8ab8150-e6f8-4eaf-987c-bcd65eac50b5",
|
||||
label = "Slack Token"
|
||||
)
|
||||
password(
|
||||
"env.YOUTRACK_TOKEN",
|
||||
"credentialsJSON:2479995b-7b60-4fbb-b095-f0bafae7f622",
|
||||
display = ParameterDisplay.HIDDEN
|
||||
)
|
||||
}
|
||||
|
||||
vcs {
|
||||
@@ -57,23 +51,23 @@ object ReleaseEap : IdeaVimBuildType({
|
||||
gradle {
|
||||
name = "Calculate new eap version"
|
||||
tasks = "scripts:calculateNewEapVersion"
|
||||
jdkHome = "/usr/lib/jvm/java-21-amazon-corretto"
|
||||
}
|
||||
gradle {
|
||||
name = "Set TeamCity build number"
|
||||
tasks = "scripts:setTeamCityBuildNumber"
|
||||
jdkHome = "/usr/lib/jvm/java-21-amazon-corretto"
|
||||
}
|
||||
gradle {
|
||||
name = "Add release tag"
|
||||
tasks = "scripts:addReleaseTag"
|
||||
jdkHome = "/usr/lib/jvm/java-21-amazon-corretto"
|
||||
}
|
||||
gradle {
|
||||
name = "Publish plugin"
|
||||
tasks = "publishPlugin"
|
||||
jdkHome = "/usr/lib/jvm/java-21-amazon-corretto"
|
||||
}
|
||||
// Same as the script below. However, jgit can't find ssh keys on TeamCity
|
||||
// gradle {
|
||||
// name = "Push changes to the repo"
|
||||
// tasks = "scripts:pushChanges"
|
||||
// }
|
||||
script {
|
||||
name = "Push changes to the repo"
|
||||
scriptContent = """
|
||||
@@ -87,11 +81,6 @@ object ReleaseEap : IdeaVimBuildType({
|
||||
git push origin %build.number%
|
||||
""".trimIndent()
|
||||
}
|
||||
gradle {
|
||||
name = "YouTrack post release actions"
|
||||
tasks = "scripts:eapReleaseActions"
|
||||
jdkHome = "/usr/lib/jvm/java-21-amazon-corretto"
|
||||
}
|
||||
}
|
||||
|
||||
features {
|
||||
|
117
.teamcity/_Self/buildTypes/ReleasePlugin.kt
vendored
117
.teamcity/_Self/buildTypes/ReleasePlugin.kt
vendored
@@ -19,6 +19,8 @@ import jetbrains.buildServer.configs.kotlin.v2019_2.ParameterDisplay
|
||||
import jetbrains.buildServer.configs.kotlin.v2019_2.buildFeatures.sshAgent
|
||||
import jetbrains.buildServer.configs.kotlin.v2019_2.buildSteps.gradle
|
||||
import jetbrains.buildServer.configs.kotlin.v2019_2.buildSteps.script
|
||||
import jetbrains.buildServer.configs.kotlin.v2019_2.failureConditions.BuildFailureOnMetric
|
||||
import jetbrains.buildServer.configs.kotlin.v2019_2.failureConditions.failOnMetricChange
|
||||
|
||||
object ReleaseMajor : ReleasePlugin("major")
|
||||
object ReleaseMinor : ReleasePlugin("minor")
|
||||
@@ -28,10 +30,7 @@ sealed class ReleasePlugin(private val releaseType: String) : IdeaVimBuildType({
|
||||
name = "Publish $releaseType release"
|
||||
description = "Build and publish IdeaVim plugin"
|
||||
|
||||
artifactRules = """
|
||||
build/distributions/*
|
||||
build/reports/*
|
||||
""".trimIndent()
|
||||
artifactRules = "build/distributions/*"
|
||||
|
||||
params {
|
||||
param("env.ORG_GRADLE_PROJECT_ideaVersion", RELEASE)
|
||||
@@ -46,11 +45,7 @@ sealed class ReleasePlugin(private val releaseType: String) : IdeaVimBuildType({
|
||||
"credentialsJSON:a8ab8150-e6f8-4eaf-987c-bcd65eac50b5",
|
||||
label = "Slack Token"
|
||||
)
|
||||
password(
|
||||
"env.ORG_GRADLE_PROJECT_youtrackToken",
|
||||
"credentialsJSON:7bc0eb3a-b86a-4ebd-b622-d4ef12d7e1d3",
|
||||
display = ParameterDisplay.HIDDEN
|
||||
)
|
||||
password("env.ORG_GRADLE_PROJECT_youtrackToken", "credentialsJSON:3cd3e867-282c-451f-b958-bc95d56a8450", display = ParameterDisplay.HIDDEN)
|
||||
param("env.ORG_GRADLE_PROJECT_releaseType", releaseType)
|
||||
}
|
||||
|
||||
@@ -70,94 +65,74 @@ sealed class ReleasePlugin(private val releaseType: String) : IdeaVimBuildType({
|
||||
name = "Pull git history"
|
||||
scriptContent = "git fetch --unshallow"
|
||||
}
|
||||
script {
|
||||
name = "Reset release branch"
|
||||
scriptContent = """
|
||||
if [ "major" = "$releaseType" ] || [ "minor" = "$releaseType" ]
|
||||
then
|
||||
echo Resetting the release branch because the release type is $releaseType
|
||||
git checkout master
|
||||
latest_eap=${'$'}(git describe --tags --match="[0-9].[0-9]*.[0-9]-eap.[0-9]*" --abbrev=0 HEAD)
|
||||
echo Latest EAP: ${'$'}latest_eap
|
||||
commit_of_latest_eap=${'$'}(git rev-list -n 1 ${'$'}latest_eap)
|
||||
echo Commit of latest EAP: ${'$'}commit_of_latest_eap
|
||||
git checkout release
|
||||
git reset --hard ${'$'}commit_of_latest_eap
|
||||
else
|
||||
git checkout release
|
||||
echo Do not reset the release branch because the release type is $releaseType
|
||||
fi
|
||||
echo Checked out release branch
|
||||
""".trimIndent()
|
||||
gradle {
|
||||
name = "Select branch"
|
||||
tasks = "scripts:selectBranch"
|
||||
}
|
||||
gradle {
|
||||
name = "Calculate new version"
|
||||
tasks = "scripts:calculateNewVersion"
|
||||
jdkHome = "/usr/lib/jvm/java-21-amazon-corretto"
|
||||
}
|
||||
gradle {
|
||||
name = "Set TeamCity build number"
|
||||
tasks = "scripts:setTeamCityBuildNumber"
|
||||
jdkHome = "/usr/lib/jvm/java-21-amazon-corretto"
|
||||
}
|
||||
// gradle {
|
||||
// name = "Update change log"
|
||||
// tasks = "scripts:changelogUpdateUnreleased"
|
||||
// }
|
||||
// gradle {
|
||||
// name = "Commit preparation changes"
|
||||
// tasks = "scripts:commitChanges"
|
||||
// }
|
||||
gradle {
|
||||
name = "Update change log"
|
||||
tasks = "scripts:changelogUpdateUnreleased"
|
||||
}
|
||||
gradle {
|
||||
name = "Commit preparation changes"
|
||||
tasks = "scripts:commitChanges"
|
||||
}
|
||||
gradle {
|
||||
name = "Add release tag"
|
||||
tasks = "scripts:addReleaseTag"
|
||||
jdkHome = "/usr/lib/jvm/java-21-amazon-corretto"
|
||||
}
|
||||
script {
|
||||
name = "Run tests"
|
||||
scriptContent = "./gradlew test -x :tests:property-tests:test -x :tests:long-running-tests:test"
|
||||
gradle {
|
||||
name = "Reset release branch"
|
||||
tasks = "scripts:resetReleaseBranch"
|
||||
}
|
||||
gradle {
|
||||
name = "Publish release"
|
||||
tasks = "publishPlugin"
|
||||
jdkHome = "/usr/lib/jvm/java-21-amazon-corretto"
|
||||
}
|
||||
// script {
|
||||
// name = "Checkout master branch"
|
||||
// scriptContent = """
|
||||
// echo Checkout master
|
||||
// git checkout master
|
||||
// """.trimIndent()
|
||||
// }
|
||||
// gradle {
|
||||
// name = "Update change log in master"
|
||||
// tasks = "scripts:changelogUpdateUnreleased"
|
||||
// }
|
||||
// gradle {
|
||||
// name = "Commit preparation changes in master"
|
||||
// tasks = "scripts:commitChanges"
|
||||
// name = "Push changes to the repo"
|
||||
// tasks = "scripts:pushChangesWithReleaseBranch"
|
||||
// }
|
||||
script {
|
||||
name = "Push changes to the repo"
|
||||
scriptContent = """
|
||||
branch=$(git branch --show-current)
|
||||
echo current branch is ${'$'}branch
|
||||
if [ "master" != "${'$'}branch" ];
|
||||
then
|
||||
git checkout master
|
||||
fi
|
||||
|
||||
git push origin --tags
|
||||
git push origin
|
||||
|
||||
if [ "patch" != $releaseType ];
|
||||
then
|
||||
git checkout release
|
||||
echo checkout release branch
|
||||
git branch --set-upstream-to=origin/release release
|
||||
git push --tags
|
||||
git push origin --force
|
||||
# Push tag
|
||||
git push origin %build.number%
|
||||
fi
|
||||
|
||||
git checkout ${'$'}branch
|
||||
""".trimIndent()
|
||||
}
|
||||
gradle {
|
||||
name = "Run Integrations"
|
||||
tasks = "releaseActions"
|
||||
gradleParams = "--no-configuration-cache"
|
||||
jdkHome = "/usr/lib/jvm/java-21-amazon-corretto"
|
||||
}
|
||||
// gradle {
|
||||
// name = "Slack Notification"
|
||||
// tasks = "slackNotification"
|
||||
// }
|
||||
gradle {
|
||||
name = "Slack Notification"
|
||||
tasks = "slackNotification"
|
||||
}
|
||||
}
|
||||
|
||||
features {
|
||||
@@ -165,4 +140,16 @@ sealed class ReleasePlugin(private val releaseType: String) : IdeaVimBuildType({
|
||||
teamcitySshKey = "IdeaVim ssh keys"
|
||||
}
|
||||
}
|
||||
|
||||
failureConditions {
|
||||
failOnMetricChange {
|
||||
metric = BuildFailureOnMetric.MetricType.ARTIFACT_SIZE
|
||||
threshold = 5
|
||||
units = BuildFailureOnMetric.MetricUnit.PERCENTS
|
||||
comparison = BuildFailureOnMetric.MetricComparison.DIFF
|
||||
compareTo = build {
|
||||
buildRule = lastSuccessful()
|
||||
}
|
||||
}
|
||||
}
|
||||
})
|
||||
|
@@ -12,7 +12,7 @@ import jetbrains.buildServer.configs.kotlin.v2019_2.triggers.vcs
|
||||
|
||||
open class TestingBuildType(
|
||||
private val testName: String,
|
||||
private val branch: String = "<default>",
|
||||
private val branch: String,
|
||||
private val version: String = testName,
|
||||
private val javaVersion: String? = null,
|
||||
private val javaPlugin: Boolean = true,
|
||||
@@ -39,11 +39,9 @@ open class TestingBuildType(
|
||||
|
||||
steps {
|
||||
gradle {
|
||||
clearConditions()
|
||||
tasks = "clean test -x :tests:property-tests:test -x :tests:long-running-tests:test"
|
||||
tasks = "clean test"
|
||||
buildFile = ""
|
||||
enableStacktrace = true
|
||||
jdkHome = "/usr/lib/jvm/java-21-amazon-corretto"
|
||||
}
|
||||
}
|
||||
|
||||
|
7
.teamcity/_Self/subprojects/GitHub.kt
vendored
7
.teamcity/_Self/subprojects/GitHub.kt
vendored
@@ -1,5 +1,6 @@
|
||||
package _Self.subprojects
|
||||
|
||||
import _Self.Constants
|
||||
import _Self.IdeaVimBuildType
|
||||
import _Self.vcsRoots.GitHubPullRequest
|
||||
import jetbrains.buildServer.configs.kotlin.v2019_2.CheckoutMode
|
||||
@@ -15,15 +16,16 @@ object GitHub : Project({
|
||||
name = "Pull Requests checks"
|
||||
description = "Automatic checking of GitHub Pull Requests"
|
||||
|
||||
buildType(GithubBuildType("clean test -x :tests:property-tests:test -x :tests:long-running-tests:test", "Tests"))
|
||||
buildType(Github("clean test", "Tests"))
|
||||
})
|
||||
|
||||
class GithubBuildType(command: String, desc: String) : IdeaVimBuildType({
|
||||
class Github(command: String, desc: String) : IdeaVimBuildType({
|
||||
name = "GitHub Pull Requests $desc"
|
||||
description = "Test GitHub pull requests $desc"
|
||||
|
||||
params {
|
||||
param("env.ORG_GRADLE_PROJECT_downloadIdeaSources", "false")
|
||||
param("env.ORG_GRADLE_PROJECT_ideaVersion", Constants.GITHUB_TESTS)
|
||||
param("env.ORG_GRADLE_PROJECT_instrumentPluginCode", "false")
|
||||
}
|
||||
|
||||
@@ -42,7 +44,6 @@ class GithubBuildType(command: String, desc: String) : IdeaVimBuildType({
|
||||
tasks = command
|
||||
buildFile = ""
|
||||
enableStacktrace = true
|
||||
jdkHome = "/usr/lib/jvm/java-21-amazon-corretto"
|
||||
}
|
||||
}
|
||||
|
||||
|
23
.teamcity/_Self/subprojects/OldTests.kt
vendored
Normal file
23
.teamcity/_Self/subprojects/OldTests.kt
vendored
Normal file
@@ -0,0 +1,23 @@
|
||||
package _Self.subprojects
|
||||
|
||||
import _Self.buildTypes.TestingBuildType
|
||||
import jetbrains.buildServer.configs.kotlin.v2019_2.Project
|
||||
|
||||
object OldTests : Project({
|
||||
name = "Old IdeaVim tests"
|
||||
description = "Tests for older versions of IJ"
|
||||
|
||||
buildType(TestingBuildType("IC-2018.1", "181-182", javaVersion = "1.8", javaPlugin = false))
|
||||
buildType(TestingBuildType("IC-2018.2", "181-182", javaVersion = "1.8", javaPlugin = false))
|
||||
buildType(TestingBuildType("IC-2018.3", "183", javaVersion = "1.8", javaPlugin = false))
|
||||
buildType(TestingBuildType("IC-2019.1", "191-193", javaVersion = "1.8", javaPlugin = false))
|
||||
buildType(TestingBuildType("IC-2019.2", "191-193", javaVersion = "1.8", javaPlugin = false))
|
||||
buildType(TestingBuildType("IC-2019.3", "191-193", javaVersion = "1.8", javaPlugin = false))
|
||||
buildType(TestingBuildType("IC-2020.1", "201", javaVersion = "1.8", javaPlugin = false))
|
||||
buildType(TestingBuildType("IC-2020.2", "202", javaVersion = "1.8", javaPlugin = false))
|
||||
buildType(TestingBuildType("IC-2020.3", "203-212", javaVersion = "1.8", javaPlugin = false))
|
||||
buildType(TestingBuildType("IC-2021.1", "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-2022.2.3", branch = "222", javaPlugin = false))
|
||||
})
|
13
.teamcity/_Self/vcsRoots/Releases.kt
vendored
13
.teamcity/_Self/vcsRoots/Releases.kt
vendored
@@ -1,13 +0,0 @@
|
||||
package _Self.vcsRoots
|
||||
|
||||
import jetbrains.buildServer.configs.kotlin.v2019_2.vcs.GitVcsRoot
|
||||
|
||||
object ReleasesVcsRoot : GitVcsRoot({
|
||||
name = "IdeaVim Releases"
|
||||
url = "git@github.com:JetBrains/ideavim.git"
|
||||
branch = "refs/heads/master"
|
||||
branchSpec = "+:refs/(*)"
|
||||
authMethod = uploadedKey {
|
||||
uploadedKey = "IdeaVim ssh keys"
|
||||
}
|
||||
})
|
20
.teamcity/patches/buildTypes/ReleasePatch.kts
vendored
Normal file
20
.teamcity/patches/buildTypes/ReleasePatch.kts
vendored
Normal file
@@ -0,0 +1,20 @@
|
||||
package patches.buildTypes
|
||||
|
||||
import jetbrains.buildServer.configs.kotlin.v2019_2.*
|
||||
import jetbrains.buildServer.configs.kotlin.v2019_2.ui.*
|
||||
|
||||
/*
|
||||
This patch script was generated by TeamCity on settings change in UI.
|
||||
To apply the patch, change the buildType with id = 'ReleasePatch'
|
||||
accordingly, and delete the patch script.
|
||||
*/
|
||||
changeBuildType(RelativeId("ReleasePatch")) {
|
||||
params {
|
||||
expect {
|
||||
password("env.ORG_GRADLE_PROJECT_youtrackToken", "credentialsJSON:3cd3e867-282c-451f-b958-bc95d56a8450", display = ParameterDisplay.HIDDEN)
|
||||
}
|
||||
update {
|
||||
password("env.ORG_GRADLE_PROJECT_youtrackToken", "credentialsJSON:7bc0eb3a-b86a-4ebd-b622-d4ef12d7e1d3", display = ParameterDisplay.HIDDEN)
|
||||
}
|
||||
}
|
||||
}
|
2
.teamcity/settings.kts
vendored
2
.teamcity/settings.kts
vendored
@@ -30,5 +30,5 @@ node (Plugins -> teamcity-configs -> teamcity-configs:generate),
|
||||
the 'Debug' option is available in the context menu for the task.
|
||||
*/
|
||||
|
||||
version = "2024.12"
|
||||
version = "2023.05"
|
||||
project(_Self.Project)
|
||||
|
171
AUTHORS.md
171
AUTHORS.md
@@ -26,13 +26,6 @@ Previous maintainers:
|
||||
|
||||
Andrey Vlasovskikh
|
||||
|
||||
Previous support members:
|
||||
|
||||
* [![icon][mail]](mailto:lejia.chen@jetbrains.com)
|
||||
[![icon][github-off]](#)
|
||||
|
||||
Lejia Chen
|
||||
|
||||
Contributors:
|
||||
|
||||
* [![icon][mail]](mailto:yole@jetbrains.com)
|
||||
@@ -490,170 +483,6 @@ Contributors:
|
||||
[![icon][github]](https://github.com/ludwig-jb)
|
||||
|
||||
ludwig-jb
|
||||
* [![icon][mail]](mailto:pvydmuch@gmail.com)
|
||||
[![icon][github]](https://github.com/pWydmuch)
|
||||
|
||||
pWydmuch
|
||||
* [![icon][mail]](mailto:leonid989@gmail.com)
|
||||
[![icon][github]](https://github.com/Infonautica)
|
||||
|
||||
Leonid Danilov
|
||||
* [![icon][mail]](mailto:emanuel-367@hotmail.com)
|
||||
[![icon][github]](https://github.com/emanuelgestosa)
|
||||
|
||||
Emanuel Gestosa
|
||||
* [![icon][mail]](mailto:81118900+lippfi@users.noreply.github.com)
|
||||
[![icon][github]](https://github.com/lippfi)
|
||||
|
||||
lippfi
|
||||
* [![icon][mail]](mailto:fillipser143@gmail.com)
|
||||
[![icon][github]](https://github.com/Parker7123)
|
||||
|
||||
FilipParker
|
||||
* [![icon][mail]](mailto:7138209+duhaesbaert@users.noreply.github.com)
|
||||
[![icon][github]](https://github.com/duhaesbaert)
|
||||
|
||||
Eduardo Haesbaert
|
||||
* [![icon][mail]](mailto:nikolaevsky.egor@gmail.com)
|
||||
[![icon][github]](https://github.com/Aisper)
|
||||
|
||||
Egor Nikolaevsky
|
||||
* [![icon][mail]](mailto:77796630+throwaway69420-69420@users.noreply.github.com)
|
||||
[![icon][github]](https://github.com/kun-codes)
|
||||
|
||||
Bishwa Saha
|
||||
* [![icon][mail]](mailto:alexfu@fastmail.com)
|
||||
[![icon][github]](https://github.com/alexfu)
|
||||
|
||||
Alex Fu
|
||||
* [![icon][mail]](mailto:jakepeters199@hotmail.com)
|
||||
[![icon][github]](https://github.com/LazyScaper)
|
||||
|
||||
Jake
|
||||
* [![icon][mail]](mailto:the1xdeveloper@gmail.com)
|
||||
[![icon][github]](https://github.com/The1xDeveloper)
|
||||
|
||||
The1xDeveloper
|
||||
* [![icon][mail]](mailto:shaunewilliams@gmail.com)
|
||||
[![icon][github]](https://github.com/shaunlebron)
|
||||
|
||||
shaun
|
||||
* [![icon][mail]](mailto:i.i.babko@gmail.com)
|
||||
[![icon][github]](https://github.com/igorbabko)
|
||||
|
||||
Igor Babko
|
||||
* [![icon][mail]](mailto:533601+felixwiemuth@users.noreply.github.com)
|
||||
[![icon][github]](https://github.com/felixwiemuth)
|
||||
|
||||
Felix Wiemuth
|
||||
* [![icon][mail]](mailto:kirill.karnaukhov@jetbrains.com)
|
||||
[![icon][github]](https://github.com/kkarnauk)
|
||||
|
||||
Kirill Karnaukhov
|
||||
* [![icon][mail]](mailto:sander.hestvik@gmail.com)
|
||||
[![icon][github]](https://github.com/SanderHestvik)
|
||||
|
||||
SanderHestvik
|
||||
* [![icon][mail]](mailto:gregory.shrago@jetbrains.com)
|
||||
[![icon][github]](https://github.com/gregsh)
|
||||
|
||||
Greg Shrago
|
||||
* [![icon][mail]](mailto:jphalip@gmail.com)
|
||||
[![icon][github]](https://github.com/jphalip)
|
||||
|
||||
Julien Phalip
|
||||
* [![icon][mail]](mailto:j.trimailovas@gmail.com)
|
||||
[![icon][github]](https://github.com/trimailov)
|
||||
|
||||
Justas Trimailovas
|
||||
* [![icon][mail]](mailto:justast@wix.com)
|
||||
[![icon][github]](https://github.com/justast-wix)
|
||||
|
||||
Justas Trimailovas
|
||||
* [![icon][mail]](mailto:wangxinhe06@gmail.com)
|
||||
[![icon][github]](https://github.com/wxh06)
|
||||
|
||||
Xinhe Wang
|
||||
* [![icon][mail]](mailto:vladimir.parfinenko@jetbrains.com)
|
||||
[![icon][github]](https://github.com/cypok)
|
||||
|
||||
Vladimir Parfinenko
|
||||
* [![icon][mail]](mailto:sdoerner@google.com)
|
||||
[![icon][github]](https://github.com/sdoerner)
|
||||
|
||||
Sebastian Dörner
|
||||
* [![icon][mail]](mailto:ocordova@pulsarml.com)
|
||||
[![icon][github]](https://github.com/oca159)
|
||||
|
||||
Osvaldo
|
||||
* [![icon][mail]](mailto:nath@squareup.com)
|
||||
[![icon][github]](https://github.com/nath)
|
||||
|
||||
Nath Tumlin
|
||||
* [![icon][mail]](mailto:ilya.usov@jetbrains.com)
|
||||
[![icon][github]](https://github.com/Iliya-usov)
|
||||
|
||||
Ilya Usov
|
||||
* [![icon][mail]](mailto:peterHoburg@users.noreply.github.com)
|
||||
[![icon][github]](https://github.com/peterHoburg)
|
||||
|
||||
Peter Hoburg
|
||||
* [![icon][mail]](mailto:erotourtes@gmail.com)
|
||||
[![icon][github]](https://github.com/erotourtes)
|
||||
|
||||
Max Siryk
|
||||
* [![icon][mail]](mailto:ivan.yarkov@jetbrains.com)
|
||||
[![icon][github]](https://github.com/MToolMakerJB)
|
||||
|
||||
Ivan Yarkov
|
||||
* [![icon][mail]](mailto:mia.vucinic@jetbrains.com)
|
||||
[![icon][github]](https://github.com/vumi19)
|
||||
|
||||
Mia Vucinic
|
||||
* [![icon][mail]](mailto:canava.thomas@gmail.com)
|
||||
[![icon][github]](https://github.com/Malandril)
|
||||
|
||||
Thomas Canava
|
||||
* [![icon][mail]](mailto:xinhe.wang@jetbrains.com)
|
||||
[![icon][github]](https://github.com/wxh06)
|
||||
|
||||
Xinhe Wang
|
||||
* [![icon][mail]](mailto:zuber.kuba@gmail.com)
|
||||
[![icon][github]](https://github.com/zuberol)
|
||||
|
||||
Jakub Zuber
|
||||
* [![icon][mail]](mailto:nmh9097@gmail.com)
|
||||
[![icon][github]](https://github.com/NaMinhyeok)
|
||||
|
||||
Na Minhyeok
|
||||
* [![icon][mail]](mailto:201638009+jetbrains-junie[bot]@users.noreply.github.com)
|
||||
[![icon][github]](https://github.com/apps/jetbrains-junie)
|
||||
|
||||
jetbrains-junie[bot]
|
||||
* [![icon][mail]](mailto:4416693+magidc@users.noreply.github.com)
|
||||
[![icon][github]](https://github.com/magidc)
|
||||
|
||||
magidc
|
||||
* [![icon][mail]](mailto:ricardo.rodcas@gmail.com)
|
||||
[![icon][github]](https://github.com/magidc)
|
||||
|
||||
magidc
|
||||
* [![icon][mail]](mailto:41898282+claude[bot]@users.noreply.github.com)
|
||||
[![icon][github]](https://github.com/apps/github-actions)
|
||||
|
||||
github-actions[bot]
|
||||
* [![icon][mail]](mailto:41898282+claude[bot]@users.noreply.github.com)
|
||||
[![icon][github]](https://github.com/apps/github-actions)
|
||||
|
||||
github-actions[bot]
|
||||
* [![icon][mail]](mailto:41898282+claude[bot]@users.noreply.github.com)
|
||||
[![icon][github]](https://github.com/apps/github-actions)
|
||||
|
||||
github-actions[bot]
|
||||
* [![icon][mail]](mailto:41898282+claude[bot]@users.noreply.github.com)
|
||||
[![icon][github]](https://github.com/apps/github-actions)
|
||||
|
||||
github-actions[bot]
|
||||
|
||||
Previous contributors:
|
||||
|
||||
|
81
CHANGES.md
81
CHANGES.md
@@ -23,92 +23,13 @@ It is important to distinguish EAP from traditional pre-release software.
|
||||
Please note that the quality of EAP versions may at times be way below even
|
||||
usual beta standards.
|
||||
|
||||
## End of changelog file maintenance
|
||||
|
||||
Since version 2.9.0, the changelog can be found on YouTrack
|
||||
|
||||
* [To Be Released](https://youtrack.jetbrains.com/issues/VIM?q=%23%7BReady%20To%20Release%7D%20)
|
||||
* [Version Fixes](https://youtrack.jetbrains.com/issues/VIM?q=State:%20Fixed%20sort%20by:%20%7BFix%20versions%7D%20asc)
|
||||
|
||||
## 2.9.0, 2024-02-20
|
||||
|
||||
### Fixes:
|
||||
* [VIM-3055](https://youtrack.jetbrains.com/issue/VIM-3055) Fix the issue with double deleting after dot
|
||||
|
||||
### Merged PRs:
|
||||
* [805](https://github.com/JetBrains/ideavim/pull/805) by [chylex](https://github.com/chylex): VIM-3238 Fix recording a macro that replays another macro
|
||||
|
||||
## 2.8.0, 2024-01-30
|
||||
|
||||
### Fixes:
|
||||
* [VIM-3130](https://youtrack.jetbrains.com/issue/VIM-3130) Change the build version to 2023.1.2
|
||||
* [VIM-3168](https://youtrack.jetbrains.com/issue/VIM-3168) Do not switch to block caret after enter if the IdeaVim is disabled
|
||||
* [VIM-3165](https://youtrack.jetbrains.com/issue/VIM-3165) Do not process enter key as IdeaVim shortcut if it's not an actual keypress
|
||||
* [VIM-3159](https://youtrack.jetbrains.com/issue/VIM-3159) Shift-enter now works in normal mode again
|
||||
* [VIM-3157](https://youtrack.jetbrains.com/issue/VIM-3157) Do not invoke enter in invokeLater for python console
|
||||
* [VIM-3195](https://youtrack.jetbrains.com/issue/VIM-3195) Fix escape in injected editor
|
||||
* [VIM-3190](https://youtrack.jetbrains.com/issue/VIM-3190) Do not use octopus handler if the enter key is used with modifiers like shift or control
|
||||
* [VIM-3203](https://youtrack.jetbrains.com/issue/VIM-3203) Split action not works in normal mode
|
||||
* [VIM-3184](https://youtrack.jetbrains.com/issue/VIM-3184) Revert "VIM-3184: Temporally disable new handlers for the thin client"
|
||||
* [VIM-3186](https://youtrack.jetbrains.com/issue/VIM-3186) Do not multiply the enter action by the amount of carets
|
||||
* [VIM-3177](https://youtrack.jetbrains.com/issue/VIM-3177) Formatting of commit message works again
|
||||
* [VIM-1611](https://youtrack.jetbrains.com/issue/VIM-1611) actions related to resolving conflicts doesn't seem to work
|
||||
* [VIM-3204](https://youtrack.jetbrains.com/issue/VIM-3204) Add checker that verifies the configuratin of the keymap
|
||||
* [VIM-3084](https://youtrack.jetbrains.com/issue/VIM-3084) Double update for the status bar icon
|
||||
* [VIM-3176](https://youtrack.jetbrains.com/issue/VIM-3176) Reselecting visual selection after pasting above it select wrong lines
|
||||
* [VIM-3206](https://youtrack.jetbrains.com/issue/VIM-3206) Disable both copilot suggestion and insert mode on a single escape
|
||||
* [VIM-3090](https://youtrack.jetbrains.com/issue/VIM-3090) Cmd line mode saves the visual mode
|
||||
* [VIM-3085](https://youtrack.jetbrains.com/issue/VIM-3085) Open access to VimTypedActionHandler and VimShortcutKeyAction
|
||||
* [VIM-3260](https://youtrack.jetbrains.com/issue/VIM-3260) Processing the offsets at the file end
|
||||
* [VIM-3183](https://youtrack.jetbrains.com/issue/VIM-3183) Execute .ideavimrc on pooled thread
|
||||
|
||||
### Merged PRs:
|
||||
* [763](https://github.com/JetBrains/ideavim/pull/763) by [Sam Ng](https://github.com/samabcde): Fix(VIM-3176) add test for restore selection after pasting in/below s…
|
||||
* [772](https://github.com/JetBrains/ideavim/pull/772) by [chylex](https://github.com/chylex): Prevent code completion popup from appearing after running a macro
|
||||
* [787](https://github.com/JetBrains/ideavim/pull/787) by [Leonid Danilov](https://github.com/Infonautica): Added "Which-Key" to Plugins
|
||||
* [778](https://github.com/JetBrains/ideavim/pull/778) by [lippfi](https://github.com/lippfi): Showmode
|
||||
* [788](https://github.com/JetBrains/ideavim/pull/788) by [Matt Ellis](https://github.com/citizenmatt): Refactor VimOptionGroupBase
|
||||
|
||||
## 2.7.0, 2023-11-07
|
||||
|
||||
### Fixes:
|
||||
* [VIM-2933](https://youtrack.jetbrains.com/issue/VIM-2933) Reloading/sourcing .ideavimrc does not initialize new plugins
|
||||
* [VIM-3138](https://youtrack.jetbrains.com/issue/VIM-3138) Do not try to register disposer if the caret is already disposed
|
||||
|
||||
### Merged PRs:
|
||||
* [734](https://github.com/JetBrains/ideavim/pull/734) by [Matt Ellis](https://github.com/citizenmatt): Support `~/` on Windows
|
||||
* [736](https://github.com/JetBrains/ideavim/pull/736) by [chylex](https://github.com/chylex): Fix(VIM-2933): Reloading/sourcing .ideavimrc does not initialize new plugins
|
||||
|
||||
## 2.6.3, 2023-10-30
|
||||
|
||||
### Changes:
|
||||
- 2.6.0 and 2.6.1 releases are broken. Version 2.6.3 reverts IdeaVim plugin to the working state as for 2.5.1.
|
||||
|
||||
## 2.6.0, 2023-10-27
|
||||
|
||||
This version of IdeaVim contains a lot of issues. Version 2.6.3 reverts these changes.
|
||||
|
||||
### Features:
|
||||
|
||||
* `ShowHoverInfo` action can be used in mappings to open a tooltip that is shown by
|
||||
mouse hovering | [VIM-2106](https://youtrack.jetbrains.com/issue/VIM-2106)
|
||||
* `has` Vim Script function supports the most common OS checks: win32, win64, linux, mac, macunix, osx, osxdarwin, bsd, sun, unix
|
||||
* See https://github.com/JetBrains/ideavim#vim-script for details about Vim Script
|
||||
## To Be Released
|
||||
|
||||
### Fixes:
|
||||
* [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:
|
||||
* [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
|
||||
|
||||
|
22
CLAUDE.md
22
CLAUDE.md
@@ -1,22 +0,0 @@
|
||||
# CLAUDE.md
|
||||
|
||||
Guidance for Claude Code when working with IdeaVim.
|
||||
|
||||
## Quick Reference
|
||||
|
||||
Essential commands:
|
||||
- `./gradlew runIde` - Start dev IntelliJ with IdeaVim
|
||||
- `./gradlew test -x :tests:property-tests:test -x :tests:long-running-tests:test` - Run standard tests
|
||||
|
||||
See CONTRIBUTING.md for architecture details and complete command list.
|
||||
|
||||
## IdeaVim-Specific Notes
|
||||
|
||||
- Property tests can be flaky - verify if failures relate to your changes
|
||||
- Use `<Action>` in mappings, not `:action`
|
||||
- Config file: `~/.ideavimrc` (XDG supported)
|
||||
- Goal: Match Vim functionality and architecture
|
||||
|
||||
## Additional Documentation
|
||||
|
||||
- Changelog maintenance: See `.claude/changelog-instructions.md`
|
@@ -1,50 +1,26 @@
|
||||
[![TeamCity Build][teamcity-build-status-svg]][teamcity-build-status]
|
||||
|
||||
IdeaVim is an open source project created by 130+ contributors. Would you like to make it even better? That’s wonderful!
|
||||
IdeaVim is an open source project created by 80+ contributors. Would you like to make it even better? That’s wonderful!
|
||||
|
||||
This page is created to help you start contributing. And who knows, maybe in a few days this project will be brighter than ever!
|
||||
|
||||
# Awards for Quality Contributions
|
||||
|
||||
In February 2025, we’re starting a program to award one-year All Products Pack subscriptions to the implementers of quality contributions to the IdeaVim project. The program will continue for all of 2025 and may be prolonged.
|
||||
|
||||
Subscriptions can be awarded for merged pull requests that meet the following requirements:
|
||||
|
||||
|
||||
- The change should be non-trivial, though there might be exceptions — for example, where a trivial fix requires a complicated investigation.
|
||||
- The change should fully implement a feature or fix the root cause of a bug. Workarounds or hacks are not accepted.
|
||||
- If applicable, the change should be properly covered with unit tests.
|
||||
- The work should be performed by the contributor, though the IdeaVim team is happy to review it and give feedback.
|
||||
- The change should fix an issue or implement a feature filed by another user. If you want to file an issue and provide a solution to it, your request for a license should be explicitly discussed with the IdeaVim team in the ticket comments.
|
||||
|
||||
|
||||
We'd like to make sure this award program is helpful and fair. Since we just started it and still fine-tuning the details, the final say on giving licenses remains with the IdeaVim team and the requirements might evolve over time.
|
||||
|
||||
|
||||
Also, a few notes:
|
||||
|
||||
|
||||
- If you have any doubts about whether your change or fix is eligible for the award, get in touch with us in the comments on YouTrack or in any other way.
|
||||
- Please mention this program in the pull request text. This is not an absolute requirement, but it will help ensure we know you would like to be considered for an award, but this is not required.
|
||||
- During 2025, a single person may only receive a single subscription. Even if you make multiple contributions, you will not be eligible for multiple awards.
|
||||
- Any delays caused by the IdeaVim team will not affect eligibility for an award if the other requirements are met.
|
||||
- Draft pull requests will not be reviewed unless explicitly requested.
|
||||
- Tickets with the [ideavim-bounty](https://youtrack.jetbrains.com/issues?q=tag:%20%7BIdeaVim-bounty%7D) tag are good candidates for this award.
|
||||
|
||||
:warning: The plugin is currently under a huge refactoring aiming to split into vim-engine and IdeaVim in order to
|
||||
support the new [Fleet IDE](https://www.jetbrains.com/fleet/). Please see [Fleet refactoring](#Fleet-refactoring).
|
||||
|
||||
## Before you begin
|
||||
|
||||
- The project is primarily written in Kotlin with a few Java files. When contributing to the project, use Kotlin unless
|
||||
you’re working in areas where Java is explicitly used.
|
||||
- The project is written in Kotlin and Java. Choose whichever language you feel more comfortable with,
|
||||
or maybe one that you’d like to get to know better (why not start [learning Kotlin](https://kotlinlang.org/docs/tutorials/) right now?).
|
||||
|
||||
- If you come across some IntelliJ Platform code, these links may prove helpful:
|
||||
|
||||
* [IntelliJ Platform SDK](https://plugins.jetbrains.com/docs/intellij/welcome.html)
|
||||
* [IntelliJ architectural overview](https://plugins.jetbrains.com/docs/intellij/fundamentals.html)
|
||||
* [IntelliJ Platform community space](https://platform.jetbrains.com/)
|
||||
* [IntelliJ architectural overview](https://www.jetbrains.org/intellij/sdk/docs/platform/fundamentals.html)
|
||||
* [IntelliJ plugin development resources](https://www.jetbrains.org/intellij/sdk/docs/welcome.html)
|
||||
|
||||
- Having any difficulties?
|
||||
Ask any questions in [GitHub discussions](https://github.com/JetBrains/ideavim/discussions) or [IntelliJ Platform community space](https://platform.jetbrains.com/).
|
||||
Join the brand new
|
||||
[](https://gitter.im/JetBrains/ideavim?utm_source=badge&utm_medium=badge&utm_campaign=pr-badge&utm_content=badge)
|
||||
for IdeaVim developers and contributors!
|
||||
|
||||
OK, ready to do some coding?
|
||||
|
||||
@@ -65,7 +41,7 @@ We've prepared some useful configurations for you:
|
||||
And here are useful gradle commands:
|
||||
|
||||
* `./gradlew runIde` — start the dev version of IntelliJ IDEA with IdeaVim installed.
|
||||
* `./gradlew test -x :tests:property-tests:test -x :tests:long-running-tests:test` — run tests.
|
||||
* `./gradlew test` — run tests.
|
||||
* `./gradlew buildPlugin` — build the plugin. The result will be located in `build/distributions`. This file can be
|
||||
installed by using `Settings | Plugin | >Gear Icon< | Install Plugin from Disk...`. You can stay with your personal build
|
||||
for a few days or send it to a friend for testing.
|
||||
@@ -78,24 +54,20 @@ for a few days or send it to a friend for testing.
|
||||
- Read the javadoc for the `@VimBehaviorDiffers` annotation in the source code and fix the corresponding functionality.
|
||||
- Implement one of the requested [#vim plugin](https://youtrack.jetbrains.com/issues/VIM?q=%23Unresolved%20tag:%20%7Bvim%20plugin%7D%20sort%20by:%20votes%20)s.
|
||||
|
||||
> :small_orange_diamond: You may leave a comment in the YouTrack ticket or open a draft PR if you’d like early feedback
|
||||
> or want to let maintainers know you’ve started working on an issue. Otherwise, simply open a PR.
|
||||
> :small_orange_diamond: Selected an issue to work on? Leave a comment in a YouTrack ticket or create a draft PR
|
||||
> to indicate that you've started working on it so that you might get additional guidance and feedback from the maintainers.
|
||||
|
||||
## Where to start in the codebase
|
||||
|
||||
If you are looking for:
|
||||
|
||||
- Vim commands (`w`, `<C-O>`, `p`, etc.):
|
||||
- Any particular command:
|
||||
- [Commands common for Fleet and IdeaVim](vim-engine/src/main/resources/ksp-generated/engine_commands.json)
|
||||
- [IdeaVim only commands](src/main/resources/ksp-generated/intellij_commands.json)
|
||||
- Any particular command: `package-info.java`.
|
||||
- How commands are executed in common: `EditorActionHandlerBase`.
|
||||
- Key mapping: `KeyHandler.handleKey()`.
|
||||
|
||||
- Ex commands (`:set`, `:s`, `:nohlsearch`):
|
||||
- Any particular command:
|
||||
- [Commands common for Fleet and IdeaVim](vim-engine/src/main/resources/ksp-generated/engine_ex_commands.json)
|
||||
- [IdeaVim only commands](src/main/resources/ksp-generated/intellij_ex_commands.json)
|
||||
- Any particular ex command: package `com.maddyhome.idea.vim.ex.handler`.
|
||||
- Vim script grammar: `Vimscript.g4`.
|
||||
- Vim script parsing: package `com.maddyhome.idea.vim.vimscript.parser`.
|
||||
- Vim script executor: `Executor`.
|
||||
@@ -106,7 +78,7 @@ If you are looking for:
|
||||
|
||||
- Common features:
|
||||
- State machine. How every particular keystroke is parsed in IdeaVim: `KeyHandler.handleKey()`.
|
||||
- Options (`incsearch`, `iskeyword`, `relativenumber`): `VimOptionGroup`.
|
||||
- Options (`incsearch`, `iskeyword`, `relativenumber`): `OptionServiceImpl`.
|
||||
- Plugin startup: `PluginStartup`.
|
||||
- Notifications: `NotificationService`.
|
||||
- Status bar icon: `StatusBar.kt`.
|
||||
@@ -130,16 +102,11 @@ Sed in orci mauris.
|
||||
Cras id tellus in ex imperdiet egestas.
|
||||
```
|
||||
|
||||
3. Don't forget to test your functionality with various corner cases:
|
||||
- **Position-based**: line start, line end, file start, file end, empty line, single character line
|
||||
- **Content-based**: whitespace-only lines, lines with trailing spaces, mixed tabs and spaces, Unicode characters, multi-byte characters (e.g., emoji, CJK)
|
||||
- **Selection-based**: multiple carets, visual mode (character/line/block), empty selection
|
||||
- **Motion-based**: dollar motion, count with motion (e.g., `3w`, `5j`), zero-width motions
|
||||
- **Buffer state**: empty file, single line file, very long lines, read-only files
|
||||
- **Boundaries**: word boundaries with punctuation, sentence/paragraph boundaries, matching brackets at extremes
|
||||
3. Don't forget to test your functionality with line start, line end, file start, file end, empty line, multiple
|
||||
carets, dollar motion, etc.
|
||||
|
||||
##### Neovim
|
||||
IdeaVim has an integration with neovim in tests. Tests that are performed with `doTest` also executed in
|
||||
IdeaVim has an experimental integration with neovim in tests. Tests that are performed with `doTest` also executed in
|
||||
neovim instance, and the state of IdeaVim is asserted to be the same as the state of neovim.
|
||||
- Only tests that use `doTest` are checked with neovim.
|
||||
- Tests with `@VimBehaviorDiffers` or `@TestWithoutNeovim` annotations don't use neovim.
|
||||
@@ -165,15 +132,14 @@ We also support proper command mappings (functions are mapped to `<Plug>...`), t
|
||||
- Magic is supported as well. See `Magic`.
|
||||
|
||||
|
||||
## Fleet
|
||||
|
||||
The IdeaVim plugin is divided into two main modules: IdeaVim and vim-engine.
|
||||
IdeaVim serves as a plugin for JetBrains IDEs, while vim-engine is an IntelliJ Platform-independent Vim engine.
|
||||
This engine is utilized in both the Vim plugin for Fleet and IdeaVim.
|
||||
## Fleet refactoring
|
||||
At the moment, IdeaVim is under an active refactoring aiming to split IdeaVim into two modules: vim-engine and IdeaVim.
|
||||
|
||||
If you develop a plugin that depends on IdeaVim: We have an instrument to check that our changes don't affect
|
||||
the plugins in the marketplace.
|
||||
the plugins in the marketplace. Also, we commit to support currently used API at least till the end of 2022.
|
||||
If you still encounter any issues with the newer versions of IdeaVim, please [contact maintainers](https://github.com/JetBrains/ideavim#contact-maintainers).
|
||||
We kindly ask you not to use anything from the new API (like `VimEditor`, `injector`) because at the moment we don't
|
||||
guarantee the compatibility of this API in the future versions.
|
||||
|
||||
|
||||
-----
|
||||
@@ -196,7 +162,6 @@ This is just terrible. [You know what to do](https://github.com/JetBrains/ideavi
|
||||
|
||||
* [Continuous integration builds](https://ideavim.teamcity.com/)
|
||||
* [Bug tracker](https://youtrack.jetbrains.com/issues/VIM)
|
||||
* [IntelliJ Platform community space](https://platform.jetbrains.com/)
|
||||
* [Chat on gitter](https://gitter.im/JetBrains/ideavim)
|
||||
* [IdeaVim Channel](https://jb.gg/bi6zp7) on [JetBrains Server](https://discord.gg/jetbrains)
|
||||
* [Plugin homepage](https://plugins.jetbrains.com/plugin/164-ideavim)
|
||||
|
60
README.md
60
README.md
@@ -29,8 +29,8 @@ IdeaVim is a Vim engine for JetBrains IDEs.
|
||||
|
||||
#### Compatibility
|
||||
|
||||
IntelliJ IDEA, PyCharm, GoLand, CLion, PhpStorm, WebStorm, RubyMine, DataGrip, DataSpell, Rider, Cursive,
|
||||
Android Studio, and other [JetBrains IDEs](https://www.jetbrains.com/ides/).
|
||||
IntelliJ IDEA, PyCharm, CLion, PhpStorm, WebStorm, RubyMine, AppCode, DataGrip, GoLand, Rider, Cursive,
|
||||
Android Studio and other IntelliJ platform based IDEs.
|
||||
|
||||
Setup
|
||||
------------
|
||||
@@ -85,16 +85,34 @@ Here are some examples of supported vim features and commands:
|
||||
* Motion / deletion / change / window / etc. commands
|
||||
* Key mappings
|
||||
* Marks / Macros / Digraphs / Registers
|
||||
* Some [set commands](https://github.com/JetBrains/ideavim/wiki/set-commands)
|
||||
* Some [set commands](https://github.com/JetBrains/ideavim/wiki/%22set%22-commands)
|
||||
* Full Vim regexps for search and search/replace
|
||||
* Vim web help
|
||||
* `~/.ideavimrc` configuration file
|
||||
* Vim script
|
||||
* IdeaVim plugins
|
||||
|
||||
[IdeaVim plugins](https://github.com/JetBrains/ideavim/wiki/IdeaVim-Plugins):
|
||||
|
||||
* vim-easymotion
|
||||
* NERDTree
|
||||
* vim-surround
|
||||
* vim-multiple-cursors
|
||||
* vim-commentary
|
||||
* argtextobj.vim
|
||||
* vim-textobj-entire
|
||||
* ReplaceWithRegister
|
||||
* vim-exchange
|
||||
* vim-highlightedyank
|
||||
* vim-paragraph-motion
|
||||
* vim-indent-object
|
||||
* match.it
|
||||
etc
|
||||
|
||||
See also:
|
||||
|
||||
* [The list of all supported commands](src/main/java/com/maddyhome/idea/vim/package-info.java)
|
||||
* [Top feature requests and bugs](https://youtrack.jetbrains.com/issues/VIM?q=%23Unresolved+sort+by%3A+votes)
|
||||
* [Vimscript support roadmap](vimscript-info/VIMSCRIPT_ROADMAP.md)
|
||||
* [List of supported in-build functions](vimscript-info/FUNCTIONS_INFO.MD)
|
||||
|
||||
Files
|
||||
-----
|
||||
@@ -204,13 +222,13 @@ Ex commands or via `:map` command mappings:
|
||||
* Execute an action by `{action_id}`. Works from Ex command line.
|
||||
* Please don't use `:action` in mappings. Use `<Action>` instead.
|
||||
|
||||
### Finding action IDs:
|
||||
### Finding action ids:
|
||||
|
||||
* IJ provides `IdeaVim: track action IDs` command to show the id of the executed actions.
|
||||
* IJ provides `IdeaVim: track action Ids` command to show the id of the executed actions.
|
||||
This command can be found in "Search everywhere" (double `shift`).
|
||||
|
||||
<details>
|
||||
<summary><strong>"Track action IDs" Details</strong> (click to see)</summary>
|
||||
<summary><strong>"Track action Ids" Details</strong> (click to see)</summary>
|
||||
<picture>
|
||||
<source media="(prefers-color-scheme: dark)" srcset="assets/readme/track_action_dark.gif">
|
||||
<img src="assets/readme/track_action_light.gif" alt="track action ids"/>
|
||||
@@ -237,7 +255,8 @@ Ex commands or via `:map` command mappings:
|
||||
##### 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
|
||||
```
|
||||
|
||||
@@ -248,7 +267,8 @@ IdeaVim can execute custom scripts that are written with Vim Script.
|
||||
At the moment we support all language features, but not all of the built-in functions and options are supported.
|
||||
|
||||
Additionally, you may be interested in the
|
||||
[Vim Script Discussion](https://github.com/JetBrains/ideavim/discussions/357).
|
||||
[Vim Script Discussion](https://github.com/JetBrains/ideavim/discussions/357) or
|
||||
[Vim Script Roadmap](https://github.com/JetBrains/ideavim/blob/master/vimscript-info/VIMSCRIPT_ROADMAP.md).
|
||||
|
||||
|
||||
### IDE specific options
|
||||
@@ -291,9 +311,7 @@ endif
|
||||
|
||||
The power of contributing drives IdeaVim :muscle:. Even small contributions matter!
|
||||
|
||||
See the contribution guide in [CONTRIBUTING.md](CONTRIBUTING.md) to start bringing your value to the project.
|
||||
|
||||
😎 In 2025, we launched a rewards program. See the guide for details.
|
||||
See [CONTRIBUTING.md](CONTRIBUTING.md) to start bringing your value to the project.
|
||||
|
||||
Authors
|
||||
-------
|
||||
@@ -307,12 +325,12 @@ IdeaVim tips and tricks
|
||||
- Use the power of IJ and Vim:
|
||||
- `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.
|
||||
- Sync IJ bookmarks and IdeaVim global marks: `set ideamarks` (works for marks with capital letters only)
|
||||
- Check out more [ex commands](https://github.com/JetBrains/ideavim/wiki/set-commands).
|
||||
- Sync IJ bookmarks and Vim marks: `set ideamarks`
|
||||
- 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`.
|
||||
- Control the status bar icon via the [`ideastatusicon` option](https://github.com/JetBrains/ideavim/wiki/set-commands).
|
||||
- Not familiar with the default behaviour during a refactoring? See the [`idearefactormode` option](https://github.com/JetBrains/ideavim/wiki/set-commands).
|
||||
- Control the status bar icon via the [`ideastatusicon` option](https://github.com/JetBrains/ideavim/wiki/%22set%22-commands).
|
||||
- Not familiar with the default behaviour during a refactoring? See the [`idearefactormode` option](https://github.com/JetBrains/ideavim/wiki/%22set%22-commands).
|
||||
|
||||
Some facts about Vim
|
||||
-------
|
||||
@@ -352,14 +370,6 @@ is the full list of synonyms.
|
||||
- Fancy constants for [undolevels](https://vimhelp.org/options.txt.html#%27undolevels%27):
|
||||
> The local value is set to -123456 when the global value is to be used.
|
||||
|
||||
- Vi (not Vim) is a POSIX standard, and [has a spec](https://pubs.opengroup.org/onlinepubs/9699919799/utilities/vi.html)! Vim is mostly POSIX compliant when Vi compatibility is selected with the `'compatible'` option, but there are still some differences that can be changed with `'copoptions'`. The spec is interesting because it documents the behaviour of different commands in a stricter style than the user documentation, describing the current line and column after the command, for example. [More details can be found by reading `:help posix`](https://vimhelp.org/vi_diff.txt.html#posix).
|
||||
|
||||
- The Vim documentation contains many easter eggs. We encounter them occasionally, but GitHub user mikesmithgh has compiled a substantial collection [here](https://github.com/mikesmithgh/vimpromptu).
|
||||
- In addition to `:call err_teapot()`, which returns `E418: I'm a teapot`, there is also `:call err_teapot(1)`, which returns `E503: Coffee is currently not available`. Naturally, this is also supported in IdeaVim.
|
||||
|
||||
- Insert mode has all `Ctrl` keys mapped, except `Ctrl-B`. In the documentation, it is marked as **"CTRL-B in Insert
|
||||
mode gone"**. Call `:h i_CTRL-B-gone` in Vim to read why `Ctrl-B` was removed.
|
||||
|
||||
License
|
||||
-------
|
||||
|
||||
|
@@ -1,5 +1,6 @@
|
||||
IdeaVim project is licensed under MIT license except the following parts of it:
|
||||
|
||||
* File [RegExp.kt](src/main/java/com/maddyhome/idea/vim/regexp/RegExp.kt) is licensed under Vim License.
|
||||
* File [ScrollViewHelper.kt](com/maddyhome/idea/vim/helper/ScrollViewHelper.kt) is licensed under Vim License.
|
||||
* File [Tutor.kt](src/main/java/com/maddyhome/idea/vim/ui/Tutor.kt) is licensed under Vim License.
|
||||
|
||||
@@ -83,8 +84,3 @@ IV) It is not allowed to remove this license from the distribution of the Vim
|
||||
license for previous Vim releases instead of the license that they came
|
||||
with, at your option.
|
||||
```
|
||||
|
||||
---
|
||||
|
||||
File [sneakIcon.png](doc/images/sneakIcon.svg), which is originally an icon of the ideavim-sneak plugin,
|
||||
is merged icons of IdeaVim plugin and a random sneaker by FreePic from flaticon.com.
|
||||
|
@@ -8,11 +8,8 @@
|
||||
|
||||
plugins {
|
||||
kotlin("jvm")
|
||||
kotlin("plugin.serialization") version "2.2.0"
|
||||
}
|
||||
|
||||
val kotlinxSerializationVersion: String by project
|
||||
|
||||
group = "com.intellij"
|
||||
version = "SNAPSHOT"
|
||||
|
||||
@@ -21,11 +18,6 @@ repositories {
|
||||
}
|
||||
|
||||
dependencies {
|
||||
compileOnly("com.google.devtools.ksp:symbol-processing-api:2.1.21-2.0.2")
|
||||
implementation("org.jetbrains.kotlinx:kotlinx-serialization-json-jvm:$kotlinxSerializationVersion") {
|
||||
// kotlin stdlib is provided by IJ, so there is no need to include it into the distribution
|
||||
exclude("org.jetbrains.kotlin", "kotlin-stdlib")
|
||||
exclude("org.jetbrains.kotlin", "kotlin-stdlib-common")
|
||||
}
|
||||
api(project(":api"))
|
||||
compileOnly("com.google.devtools.ksp:symbol-processing-api:1.9.10-1.0.13")
|
||||
implementation("org.jetbrains.kotlinx:kotlinx-serialization-json-jvm:1.6.0")
|
||||
}
|
@@ -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,66 +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> {
|
||||
val commandsFile = environment.options["commands_file"]
|
||||
if (commandsFile == null) return emptyList()
|
||||
|
||||
resolver.getAllFiles().forEach { it.accept(visitor, Unit) }
|
||||
|
||||
val generatedDirPath = Path(environment.options["generated_directory"]!!)
|
||||
Files.createDirectories(generatedDirPath)
|
||||
|
||||
val filePath = generatedDirPath.resolve(commandsFile)
|
||||
val sortedCommands = commands.sortedWith(compareBy({ it.keys }, { it.`class` }))
|
||||
val fileContent = json.encodeToString(sortedCommands)
|
||||
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 kotlinx.serialization.encodeToString
|
||||
import kotlinx.serialization.json.Json
|
||||
import java.nio.file.Files
|
||||
import kotlin.io.path.Path
|
||||
import kotlin.io.path.writeText
|
||||
|
||||
@@ -31,17 +30,9 @@ class ExCommandProcessor(private val environment: SymbolProcessorEnvironment): S
|
||||
private val json = Json { prettyPrint = true }
|
||||
|
||||
override fun process(resolver: Resolver): List<KSAnnotated> {
|
||||
val exCommandsFile = environment.options["ex_commands_file"]
|
||||
if (exCommandsFile == null) return emptyList()
|
||||
|
||||
resolver.getAllFiles().forEach { it.accept(visitor, Unit) }
|
||||
|
||||
val generatedDirPath = Path(environment.options["generated_directory"]!!)
|
||||
Files.createDirectories(generatedDirPath)
|
||||
|
||||
val filePath = generatedDirPath.resolve(exCommandsFile)
|
||||
val sortedCommandToClass = commandToClass.toList().sortedWith(compareBy({ it.first }, { it.second })).toMap()
|
||||
val fileContent = json.encodeToString(sortedCommandToClass)
|
||||
val filePath = Path(environment.options["generated_directory"]!!, environment.options["ex_commands_file"]!!)
|
||||
val fileContent = json.encodeToString(commandToClass)
|
||||
filePath.writeText(fileContent)
|
||||
|
||||
return emptyList()
|
||||
|
@@ -1,78 +0,0 @@
|
||||
/*
|
||||
* Copyright 2003-2025 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.KSFunctionDeclaration
|
||||
import com.google.devtools.ksp.symbol.KSVisitorVoid
|
||||
import com.intellij.vim.api.VimPlugin
|
||||
import kotlinx.serialization.encodeToString
|
||||
import kotlinx.serialization.json.Json
|
||||
import java.nio.file.Files
|
||||
import kotlin.io.path.Path
|
||||
import kotlin.io.path.writeText
|
||||
|
||||
// Used for processing VimPlugin annotations
|
||||
class ExtensionsProcessor(private val environment: SymbolProcessorEnvironment) : SymbolProcessor {
|
||||
private val visitor = ExtensionsVisitor()
|
||||
private val declaredExtensions = mutableListOf<KspExtensionBean>()
|
||||
|
||||
private val json = Json { prettyPrint = true }
|
||||
|
||||
override fun process(resolver: Resolver): List<KSAnnotated> {
|
||||
val extensionsFile = environment.options["extensions_file"]
|
||||
if (extensionsFile == null) return emptyList()
|
||||
|
||||
resolver.getAllFiles().forEach { it.accept(visitor, Unit) }
|
||||
|
||||
val generatedDirPath = Path(environment.options["generated_directory"]!!)
|
||||
Files.createDirectories(generatedDirPath)
|
||||
|
||||
val filePath = generatedDirPath.resolve(environment.options["extensions_file"]!!)
|
||||
val sortedExtensions = declaredExtensions.toList().sortedWith(compareBy { it.extensionName })
|
||||
|
||||
val fileContent = json.encodeToString(sortedExtensions)
|
||||
filePath.writeText(fileContent)
|
||||
|
||||
return emptyList()
|
||||
}
|
||||
|
||||
|
||||
private inner class ExtensionsVisitor : KSVisitorVoid() {
|
||||
@OptIn(KspExperimental::class)
|
||||
override fun visitFunctionDeclaration(function: KSFunctionDeclaration, data: Unit) {
|
||||
val pluginAnnotation = function.getAnnotationsByType(VimPlugin::class).firstOrNull() ?: return
|
||||
val pluginName = pluginAnnotation.name
|
||||
val functionPath = function.simpleName.asString()
|
||||
|
||||
// Extensions are not declared as part of class, however, when Kotlin code is decompiled to Java,
|
||||
// function `fun init()` in a file File.kt, will be a static method in a class FileKt since Java
|
||||
// does not support top-level functions. Then, it can be loaded with class loader.
|
||||
|
||||
val surroundingFileName = function.containingFile?.fileName?.removeSuffix(".kt") ?: return
|
||||
val packageName = function.packageName.asString()
|
||||
|
||||
val className = "$packageName.${surroundingFileName}Kt"
|
||||
|
||||
val kspExtensionBean = KspExtensionBean(pluginName, functionPath, className)
|
||||
declaredExtensions.add(kspExtensionBean)
|
||||
}
|
||||
|
||||
override fun visitFile(file: KSFile, data: Unit) {
|
||||
file.declarations.forEach { it.accept(this, Unit) }
|
||||
}
|
||||
}
|
||||
}
|
@@ -1,14 +0,0 @@
|
||||
/*
|
||||
* Copyright 2003-2025 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 KspExtensionBean(val extensionName: String, val functionName: String, val className: String)
|
@@ -20,7 +20,6 @@ import com.google.devtools.ksp.symbol.KSVisitorVoid
|
||||
import com.intellij.vim.annotations.VimscriptFunction
|
||||
import kotlinx.serialization.encodeToString
|
||||
import kotlinx.serialization.json.Json
|
||||
import java.nio.file.Files
|
||||
import kotlin.io.path.Path
|
||||
import kotlin.io.path.writeText
|
||||
|
||||
@@ -31,17 +30,9 @@ class VimscriptFunctionProcessor(private val environment: SymbolProcessorEnviron
|
||||
private val json = Json { prettyPrint = true }
|
||||
|
||||
override fun process(resolver: Resolver): List<KSAnnotated> {
|
||||
val vimscriptFunctionsFile = environment.options["vimscript_functions_file"]
|
||||
if (vimscriptFunctionsFile == null) return emptyList()
|
||||
|
||||
resolver.getAllFiles().forEach { it.accept(visitor, Unit) }
|
||||
|
||||
val generatedDirPath = Path(environment.options["generated_directory"]!!)
|
||||
Files.createDirectories(generatedDirPath)
|
||||
|
||||
val filePath = generatedDirPath.resolve(vimscriptFunctionsFile)
|
||||
val sortedNameToClass = nameToClass.toList().sortedWith(compareBy({ it.first }, { it.second })).toMap()
|
||||
val fileContent = json.encodeToString(sortedNameToClass)
|
||||
val filePath = Path(environment.options["generated_directory"]!!, environment.options["vimscript_functions_file"]!!)
|
||||
val fileContent = json.encodeToString(nameToClass)
|
||||
filePath.writeText(fileContent)
|
||||
|
||||
return emptyList()
|
||||
|
@@ -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,20 +0,0 @@
|
||||
/*
|
||||
* Copyright 2003-2025 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.ExtensionsProcessor
|
||||
|
||||
class ExtensionsProcessorProvider: SymbolProcessorProvider {
|
||||
override fun create(environment: SymbolProcessorEnvironment): SymbolProcessor {
|
||||
return ExtensionsProcessor(environment)
|
||||
}
|
||||
}
|
@@ -13,7 +13,7 @@ import com.google.devtools.ksp.processing.SymbolProcessorEnvironment
|
||||
import com.google.devtools.ksp.processing.SymbolProcessorProvider
|
||||
import com.intellij.vim.processors.VimscriptFunctionProcessor
|
||||
|
||||
class VimscriptFunctionProcessorProvider : SymbolProcessorProvider {
|
||||
public class VimscriptFunctionProcessorProvider : SymbolProcessorProvider {
|
||||
override fun create(environment: SymbolProcessorEnvironment): SymbolProcessor {
|
||||
return VimscriptFunctionProcessor(environment)
|
||||
}
|
||||
|
@@ -1,4 +1,2 @@
|
||||
com.intellij.vim.providers.CommandOrMotionProcessorProvider
|
||||
com.intellij.vim.providers.ExCommandProcessorProvider
|
||||
com.intellij.vim.providers.VimscriptFunctionProcessorProvider
|
||||
com.intellij.vim.providers.ExtensionsProcessorProvider
|
||||
com.intellij.vim.providers.ExCommandProcessorProvider
|
@@ -1,29 +0,0 @@
|
||||
/*
|
||||
* Copyright 2003-2025 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.
|
||||
*/
|
||||
|
||||
plugins {
|
||||
kotlin("jvm")
|
||||
}
|
||||
|
||||
val kotlinVersion: String by project
|
||||
|
||||
repositories {
|
||||
mavenCentral()
|
||||
}
|
||||
|
||||
dependencies {
|
||||
testImplementation(platform("org.junit:junit-bom:6.0.0"))
|
||||
testImplementation("org.junit.jupiter:junit-jupiter")
|
||||
compileOnly("org.jetbrains.kotlin:kotlin-stdlib:$kotlinVersion")
|
||||
compileOnly("org.jetbrains:annotations:26.0.2-1")
|
||||
compileOnly("org.jetbrains.kotlinx:kotlinx-coroutines-core-jvm:1.10.2")
|
||||
}
|
||||
|
||||
tasks.test {
|
||||
useJUnitPlatform()
|
||||
}
|
@@ -1,545 +0,0 @@
|
||||
/*
|
||||
* Copyright 2003-2025 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.api
|
||||
|
||||
import com.intellij.vim.api.models.Mode
|
||||
import com.intellij.vim.api.models.Path
|
||||
import com.intellij.vim.api.scopes.DigraphScope
|
||||
import com.intellij.vim.api.scopes.MappingScope
|
||||
import com.intellij.vim.api.scopes.ModalInput
|
||||
import com.intellij.vim.api.scopes.OptionScope
|
||||
import com.intellij.vim.api.scopes.OutputPanelScope
|
||||
import com.intellij.vim.api.scopes.VimApiDsl
|
||||
import com.intellij.vim.api.scopes.commandline.CommandLineScope
|
||||
import com.intellij.vim.api.scopes.editor.EditorScope
|
||||
import org.jetbrains.annotations.ApiStatus
|
||||
import kotlin.reflect.KType
|
||||
import kotlin.reflect.typeOf
|
||||
|
||||
/**
|
||||
* Entry point of the Vim API
|
||||
*
|
||||
* The API is currently in experimental status and not suggested to be used.
|
||||
*/
|
||||
@ApiStatus.Experimental
|
||||
@VimApiDsl
|
||||
interface VimApi {
|
||||
/**
|
||||
* Represents the current mode in Vim.
|
||||
*
|
||||
* Example usage:
|
||||
*
|
||||
* **Getting the Current Mode**
|
||||
* ```kotlin
|
||||
* val currentMode = mode
|
||||
* println("Current Vim Mode: $currentMode")
|
||||
* ```
|
||||
*
|
||||
* The set of mode is currently an experimental operation as the contracts of it are getting polished.
|
||||
* We suggest currently not using it.
|
||||
*/
|
||||
@set:ApiStatus.Experimental
|
||||
var mode: Mode
|
||||
|
||||
/**
|
||||
* Retrieves a variable of the specified type and name.
|
||||
* Use the extension function `getVariable<String>("name")`
|
||||
*/
|
||||
fun <T : Any> getVariable(name: String, type: KType): T?
|
||||
|
||||
/**
|
||||
* Sets a variable with the specified name and value.
|
||||
* Use the extension function `setVariable<String>("name", 1)`
|
||||
*
|
||||
* In Vim, this is equivalent to `let varname = value`.
|
||||
*/
|
||||
fun setVariable(name: String, value: Any, type: KType)
|
||||
|
||||
/**
|
||||
* Exports a function that can be used as an operator function in Vim.
|
||||
*
|
||||
* In Vim, operator functions are used with the `g@` operator to create custom operators.
|
||||
*
|
||||
* Example usage:
|
||||
* ```kotlin
|
||||
* exportOperatorFunction("MyOperator") {
|
||||
* editor {
|
||||
* // Perform operations on the selected text
|
||||
* true // Return success
|
||||
* }
|
||||
* }
|
||||
* ```
|
||||
*
|
||||
* @param name The name to register the function under
|
||||
* @param function The function to execute when the operator is invoked
|
||||
*/
|
||||
fun exportOperatorFunction(name: String, function: suspend VimApi.() -> Boolean)
|
||||
|
||||
/**
|
||||
* Sets the current operator function to use with the `g@` operator.
|
||||
*
|
||||
* In Vim, this is equivalent to setting the 'operatorfunc' option.
|
||||
*
|
||||
* @param name The name of the previously exported operator function
|
||||
*/
|
||||
fun setOperatorFunction(name: String)
|
||||
|
||||
/**
|
||||
* Executes normal mode commands as if they were typed.
|
||||
*
|
||||
* In Vim, this is equivalent to the `:normal` command.
|
||||
*
|
||||
* Example usage:
|
||||
* ```kotlin
|
||||
* normal("gg") // Go to the first line
|
||||
* normal("dw") // Delete word
|
||||
* ```
|
||||
*
|
||||
* @param command The normal mode command string to execute
|
||||
*/
|
||||
fun normal(command: String)
|
||||
|
||||
/**
|
||||
* Executes a block of code in the context of the currently focused editor.
|
||||
*
|
||||
* Example usage:
|
||||
* ```kotlin
|
||||
* editor {
|
||||
* read {
|
||||
* // executed under read lock
|
||||
* }
|
||||
* }
|
||||
* ```
|
||||
*
|
||||
* @param block The code block to execute within editor scope
|
||||
* @return The result of the block execution
|
||||
*/
|
||||
fun <T> editor(block: EditorScope.() -> T): T
|
||||
|
||||
/**
|
||||
* Executes a block of code for each editor.
|
||||
*
|
||||
* This function allows performing operations on all available editors.
|
||||
*
|
||||
* Example usage:
|
||||
* ```kotlin
|
||||
* forEachEditor {
|
||||
* // Perform some operation on each editor
|
||||
* }
|
||||
* ```
|
||||
*
|
||||
* @param block The code block to execute for each editor
|
||||
* @return A list containing the results of executing the block on each editor
|
||||
*/
|
||||
fun <T> forEachEditor(block: EditorScope.() -> T): List<T>
|
||||
|
||||
/**
|
||||
* Provides access to key mapping functionality.
|
||||
*
|
||||
* Example usage:
|
||||
* ```kotlin
|
||||
* mappings {
|
||||
* nmap("jk", "<Esc>")
|
||||
* }
|
||||
* ```
|
||||
*
|
||||
* @param block The code block to execute within the mapping scope
|
||||
*/
|
||||
fun mappings(block: MappingScope.() -> Unit)
|
||||
|
||||
// /**
|
||||
// * Provides access to event listener functionality.
|
||||
// *
|
||||
// * Example usage:
|
||||
// * ```kotlin
|
||||
// * listeners {
|
||||
// * // Register a listener for mode changes
|
||||
// * onModeChange { oldMode ->
|
||||
// * println("Mode changed from $oldMode")
|
||||
// * }
|
||||
// * }
|
||||
// * ```
|
||||
// *
|
||||
// * @param block The code block to execute within the listeners scope
|
||||
// */
|
||||
// fun listeners(block: ListenersScope.() -> Unit)
|
||||
|
||||
/**
|
||||
* Provides access to Vim's output panel functionality.
|
||||
*
|
||||
* Example usage:
|
||||
* ```kotlin
|
||||
* outputPanel {
|
||||
* // Print a message to the output panel
|
||||
* setText("Hello from IdeaVim plugin!")
|
||||
* }
|
||||
* ```
|
||||
*
|
||||
* @param block The code block to execute within the output panel scope
|
||||
*/
|
||||
fun outputPanel(block: OutputPanelScope.() -> Unit)
|
||||
|
||||
/**
|
||||
* Provides access to modal input functionality.
|
||||
*
|
||||
* Example usage:
|
||||
* ```kotlin
|
||||
* modalInput()
|
||||
* .inputChar(label) { char ->
|
||||
* // get char that user entered
|
||||
* }
|
||||
* ```
|
||||
*
|
||||
* @return A ModalInput instance that can be used to request user input
|
||||
*/
|
||||
fun modalInput(): ModalInput
|
||||
|
||||
/**
|
||||
* Provides access to Vim's command line functionality.
|
||||
*
|
||||
* Example usage:
|
||||
* ```kotlin
|
||||
* commandLine {
|
||||
* // get current command line text
|
||||
* read {
|
||||
* // executed under read lock
|
||||
* text
|
||||
* }
|
||||
* }
|
||||
* ```
|
||||
*
|
||||
* @param block The code block to execute with command line scope
|
||||
*/
|
||||
fun commandLine(block: CommandLineScope.() -> Unit)
|
||||
|
||||
/**
|
||||
* Provides access to Vim's options functionality.
|
||||
*
|
||||
* Example usage:
|
||||
* ```kotlin
|
||||
* // Get option value
|
||||
* val history = option { get<Int>("history") }
|
||||
*
|
||||
* // Set option value and return result
|
||||
* val wasSet = option {
|
||||
* set("number", true)
|
||||
* true
|
||||
* }
|
||||
*
|
||||
* // Multiple operations
|
||||
* option {
|
||||
* set("ignorecase", true)
|
||||
* append("virtualedit", "block")
|
||||
* }
|
||||
* ```
|
||||
*
|
||||
* @param block The code block to execute within the option scope
|
||||
* @return The result of the block execution
|
||||
*/
|
||||
fun <T> option(block: OptionScope.() -> T): T
|
||||
|
||||
/**
|
||||
* Provides access to Vim's digraph functionality.
|
||||
*
|
||||
* Example usage:
|
||||
* ```kotlin
|
||||
* digraph {
|
||||
* // Add a new digraph
|
||||
* add("a:", 'ä')
|
||||
* }
|
||||
* ```
|
||||
*
|
||||
* @param block The code block to execute within the digraph scope
|
||||
*/
|
||||
fun digraph(block: DigraphScope.() -> Unit)
|
||||
|
||||
/**
|
||||
* Gets the number of tabs in the current window.
|
||||
*/
|
||||
val tabCount: Int
|
||||
|
||||
/**
|
||||
* The index of the current tab or null if there is no tab selected or no tabs are open
|
||||
*/
|
||||
val currentTabIndex: Int?
|
||||
|
||||
/**
|
||||
* Removes a tab at the specified index and selects another tab.
|
||||
*
|
||||
* @param indexToDelete The index of the tab to delete
|
||||
* @param indexToSelect The index of the tab to select after deletion
|
||||
*/
|
||||
fun removeTabAt(indexToDelete: Int, indexToSelect: Int)
|
||||
|
||||
/**
|
||||
* Moves the current tab to the specified index.
|
||||
*
|
||||
* @param index The index to move the current tab to
|
||||
* @throws IllegalStateException if there is no tab selected or no tabs are open
|
||||
*/
|
||||
fun moveCurrentTabToIndex(index: Int)
|
||||
|
||||
/**
|
||||
* Closes all tabs except the current one.
|
||||
*
|
||||
* @throws IllegalStateException if there is no tab selected
|
||||
*/
|
||||
fun closeAllExceptCurrentTab()
|
||||
|
||||
/**
|
||||
* Checks if a pattern matches a text.
|
||||
*
|
||||
* @param pattern The regular expression pattern to match
|
||||
* @param text The text to check against the pattern
|
||||
* @param ignoreCase Whether to ignore case when matching
|
||||
* @return True if the pattern matches the text, false otherwise
|
||||
*/
|
||||
fun matches(pattern: String, text: String, ignoreCase: Boolean = false): Boolean
|
||||
|
||||
/**
|
||||
* Finds all matches of a pattern in a text.
|
||||
*
|
||||
* @param text The text to search in
|
||||
* @param pattern The regular expression pattern to search for
|
||||
* @return A list of pairs representing the start and end offsets of each match
|
||||
*/
|
||||
fun getAllMatches(text: String, pattern: String): List<Pair<Int, Int>>
|
||||
|
||||
/**
|
||||
* Selects the next window in the editor.
|
||||
*/
|
||||
fun selectNextWindow()
|
||||
|
||||
/**
|
||||
* Selects the previous window in the editor.
|
||||
*/
|
||||
fun selectPreviousWindow()
|
||||
|
||||
/**
|
||||
* Selects a window by its index.
|
||||
*
|
||||
* @param index The index of the window to select (1-based).
|
||||
*/
|
||||
fun selectWindow(index: Int)
|
||||
|
||||
/**
|
||||
* Splits the current window vertically and optionally opens a file in the new window.
|
||||
*
|
||||
* @param filePath Path of the file to open in the new window. If null, the new window will show the same file.
|
||||
*/
|
||||
fun splitWindowVertically(filePath: Path? = null)
|
||||
|
||||
/**
|
||||
* Splits the current window horizontally and optionally opens a file in the new window.
|
||||
*
|
||||
* @param filePath Path of the file to open in the new window. If null, the new window will show the same file.
|
||||
*/
|
||||
fun splitWindowHorizontally(filePath: Path? = null)
|
||||
|
||||
/**
|
||||
* Closes all windows except the current one.
|
||||
*/
|
||||
fun closeAllExceptCurrentWindow()
|
||||
|
||||
/**
|
||||
* Closes the current window.
|
||||
*/
|
||||
fun closeCurrentWindow()
|
||||
|
||||
/**
|
||||
* Closes all windows in the editor.
|
||||
*/
|
||||
fun closeAllWindows()
|
||||
|
||||
/**
|
||||
* Parses and executes the given Vimscript string.
|
||||
*
|
||||
* @param script The Vimscript string to execute
|
||||
* @return The result of the execution, which can be Success or Error
|
||||
*/
|
||||
fun execute(script: String): Boolean
|
||||
|
||||
/**
|
||||
* Registers a new Vim command.
|
||||
*
|
||||
* Example usage:
|
||||
* ```
|
||||
* command("MyCommand") { cmd ->
|
||||
* println("Command executed: $cmd")
|
||||
* }
|
||||
* ```
|
||||
*
|
||||
* @param command The name of the command to register, as entered by the user.
|
||||
* @param block The logic to execute when the command is invoked. Receives the command name
|
||||
* entered by the user as a parameter.
|
||||
*/
|
||||
fun command(command: String, block: VimApi.(String) -> Unit)
|
||||
|
||||
/**
|
||||
* Gets keyed data from a Vim window.
|
||||
*
|
||||
* @param key The key to retrieve data for
|
||||
* @return The data associated with the key, or null if no data is found
|
||||
*/
|
||||
fun <T> getDataFromWindow(key: String): T?
|
||||
|
||||
/**
|
||||
* Stores keyed user data in a Vim window.
|
||||
*
|
||||
* @param key The key to store data for
|
||||
* @param data The data to store
|
||||
*/
|
||||
fun <T> putDataToWindow(key: String, data: T)
|
||||
|
||||
/**
|
||||
* Gets data from buffer.
|
||||
*
|
||||
* @param key The key to retrieve data for
|
||||
* @return The data associated with the key, or null if no data is found
|
||||
*/
|
||||
fun <T> getDataFromBuffer(key: String): T?
|
||||
|
||||
/**
|
||||
* Puts data to buffer.
|
||||
*
|
||||
* @param key The key to store data for
|
||||
* @param data The data to store
|
||||
*/
|
||||
fun <T> putDataToBuffer(key: String, data: T)
|
||||
|
||||
/**
|
||||
* Gets data from tab (group of windows).
|
||||
*
|
||||
* @param key The key to retrieve data for
|
||||
* @return The data associated with the key, or null if no data is found
|
||||
*/
|
||||
fun <T> getDataFromTab(key: String): T?
|
||||
|
||||
/**
|
||||
* Puts data to tab (group of windows).
|
||||
*
|
||||
* @param key The key to store data for
|
||||
* @param data The data to store
|
||||
*/
|
||||
fun <T> putDataToTab(key: String, data: T)
|
||||
|
||||
/**
|
||||
* Gets data from window or puts it if it doesn't exist.
|
||||
*
|
||||
* @param key The key to retrieve or store data for
|
||||
* @param provider A function that provides the data if it doesn't exist
|
||||
* @return The existing data or the newly created data
|
||||
*/
|
||||
fun <T> getOrPutWindowData(key: String, provider: () -> T): T =
|
||||
getDataFromWindow(key) ?: provider().also { putDataToWindow(key, it) }
|
||||
|
||||
/**
|
||||
* Gets data from buffer or puts it if it doesn't exist.
|
||||
*
|
||||
* @param key The key to retrieve or store data for
|
||||
* @param provider A function that provides the data if it doesn't exist
|
||||
* @return The existing data or the newly created data
|
||||
*/
|
||||
fun <T> getOrPutBufferData(key: String, provider: () -> T): T =
|
||||
getDataFromBuffer(key) ?: provider().also { putDataToBuffer(key, it) }
|
||||
|
||||
/**
|
||||
* Gets data from tab or puts it if it doesn't exist.
|
||||
*
|
||||
* @param key The key to retrieve or store data for
|
||||
* @param provider A function that provides the data if it doesn't exist
|
||||
* @return The existing data or the newly created data
|
||||
*/
|
||||
fun <T> getOrPutTabData(key: String, provider: () -> T): T =
|
||||
getDataFromTab(key) ?: provider().also { putDataToTab(key, it) }
|
||||
|
||||
/**
|
||||
* Saves the current file.
|
||||
*/
|
||||
fun saveFile()
|
||||
|
||||
/**
|
||||
* Closes the current file.
|
||||
*/
|
||||
fun closeFile()
|
||||
|
||||
/**
|
||||
* Finds the start offset of the next word in camel case or snake case text.
|
||||
*
|
||||
* @param chars The character sequence to search in (e.g., document text)
|
||||
* @param startIndex The index to start searching from (inclusive). Must be within the bounds of chars: [0, chars.length)
|
||||
* @param count Find the [count]-th occurrence. Must be greater than 0.
|
||||
* @return The offset of the next word start, or null if not found
|
||||
*/
|
||||
fun getNextCamelStartOffset(chars: CharSequence, startIndex: Int, count: Int = 1): Int?
|
||||
|
||||
/**
|
||||
* Finds the start offset of the previous word in camel case or snake case text.
|
||||
*
|
||||
* @param chars The character sequence to search in (e.g., document text)
|
||||
* @param endIndex The index to start searching backward from (exclusive). Must be within the bounds of chars: [0, chars.length]
|
||||
* @param count Find the [count]-th occurrence. Must be greater than 0.
|
||||
* @return The offset of the previous word start, or null if not found
|
||||
*/
|
||||
fun getPreviousCamelStartOffset(chars: CharSequence, endIndex: Int, count: Int = 1): Int?
|
||||
|
||||
/**
|
||||
* Finds the end offset of the next word in camel case or snake case text.
|
||||
*
|
||||
* @param chars The character sequence to search in (e.g., document text)
|
||||
* @param startIndex The index to start searching from (inclusive). Must be within the bounds of chars: [0, chars.length)
|
||||
* @param count Find the [count]-th occurrence. Must be greater than 0.
|
||||
* @return The offset of the next word end, or null if not found
|
||||
*/
|
||||
fun getNextCamelEndOffset(chars: CharSequence, startIndex: Int, count: Int = 1): Int?
|
||||
|
||||
/**
|
||||
* Finds the end offset of the previous word in camel case or snake case text.
|
||||
*
|
||||
* @param chars The character sequence to search in (e.g., document text)
|
||||
* @param endIndex The index to start searching backward from (exclusive). Must be within the bounds of chars: [0, chars.length]
|
||||
* @param count Find the [count]-th occurrence. Must be greater than 0.
|
||||
* @return The offset of the previous word end, or null if not found
|
||||
*/
|
||||
fun getPreviousCamelEndOffset(chars: CharSequence, endIndex: Int, count: Int = 1): Int?
|
||||
}
|
||||
|
||||
/**
|
||||
* Sets a variable with the specified name and value.
|
||||
*
|
||||
* In Vim, this is equivalent to `let varname = value`.
|
||||
*
|
||||
* Example usage:
|
||||
* ```
|
||||
* setVariable<Int>("g:my_var", 42)
|
||||
* ```
|
||||
*
|
||||
* @param name The name of the variable, optionally prefixed with a scope (g:, b:, etc.)
|
||||
* @param value The value to set
|
||||
*/
|
||||
inline fun <reified T : Any> VimApi.setVariable(name: String, value: T) {
|
||||
val kType: KType = typeOf<T>()
|
||||
setVariable(name, value, kType)
|
||||
}
|
||||
|
||||
/**
|
||||
* Retrieves a variable of the specified type and name.
|
||||
*
|
||||
* Example usage:
|
||||
* ```
|
||||
* val value: String? = getVariable<String>("myVariable")
|
||||
* ```
|
||||
*
|
||||
* @param name The name of the variable to retrieve.
|
||||
* @return The variable of type `T` if found, otherwise `null`.
|
||||
*/
|
||||
inline fun <reified T : Any> VimApi.getVariable(name: String): T? {
|
||||
val kType: KType = typeOf<T>()
|
||||
return getVariable(name, kType)
|
||||
}
|
@@ -1,18 +0,0 @@
|
||||
/*
|
||||
* Copyright 2003-2025 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.api
|
||||
|
||||
/**
|
||||
* Annotation used to describe a Vim plugin.
|
||||
*
|
||||
* @property name Specifies the name of the plugin.
|
||||
*/
|
||||
@Target(AnnotationTarget.FUNCTION)
|
||||
@Retention(AnnotationRetention.RUNTIME)
|
||||
annotation class VimPlugin(val name: String)
|
@@ -1,70 +0,0 @@
|
||||
/*
|
||||
* Copyright 2003-2025 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.api.models
|
||||
|
||||
/**
|
||||
* Represents the type of text selection in Vim.
|
||||
*/
|
||||
enum class TextType {
|
||||
/**
|
||||
* Character-wise selection, where text is selected character by character.
|
||||
*/
|
||||
CHARACTER_WISE,
|
||||
|
||||
/**
|
||||
* Line-wise selection, where text is selected line by line.
|
||||
*/
|
||||
LINE_WISE,
|
||||
|
||||
/**
|
||||
* Block-wise selection, where text is selected in a rectangular block.
|
||||
*/
|
||||
BLOCK_WISE,
|
||||
}
|
||||
|
||||
/**
|
||||
* Represents a line of text in the editor.
|
||||
*
|
||||
* @property number The line number (0-based or 1-based depending on context).
|
||||
* @property text The content of the line.
|
||||
* @property start The offset of the first character in the line.
|
||||
* @property end The offset after the last character in the line.
|
||||
*/
|
||||
data class Line(val number: Int, val text: String, val start: Int, val end: Int)
|
||||
|
||||
/**
|
||||
* Represents a caret with its associated information.
|
||||
* A pair of [CaretId] and [CaretInfo].
|
||||
*/
|
||||
typealias CaretData = Pair<CaretId, CaretInfo>
|
||||
|
||||
/**
|
||||
* A unique identifier for a caret in the editor.
|
||||
*
|
||||
* @property id The string representation of the caret identifier.
|
||||
*/
|
||||
@JvmInline
|
||||
value class CaretId(val id: String)
|
||||
|
||||
/**
|
||||
* Contains information about a caret's position and selection.
|
||||
*
|
||||
* @property offset The current offset (position) of the caret in the document.
|
||||
* @property selection The selection range as a pair of start and end offsets, or null if no selection.
|
||||
*/
|
||||
data class CaretInfo(
|
||||
val offset: Int,
|
||||
val selection: Pair<Int, Int>?,
|
||||
)
|
||||
|
||||
/**
|
||||
* Represents an identifier for a highlight in the editor.
|
||||
* Used for tracking and managing highlights applied to text.
|
||||
*/
|
||||
interface HighlightId
|
@@ -1,55 +0,0 @@
|
||||
/*
|
||||
* Copyright 2003-2025 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.api.models
|
||||
|
||||
/**
|
||||
* Represents a color in RGBA format.
|
||||
*
|
||||
* @property hexCode The string representation of the color in hex format (#RRGGBB or #RRGGBBAA).
|
||||
*/
|
||||
data class Color(
|
||||
val hexCode: String,
|
||||
) {
|
||||
/**
|
||||
* Creates a color from individual RGB(A) components.
|
||||
*
|
||||
* @param r Red component (0-255).
|
||||
* @param g Green component (0-255).
|
||||
* @param b Blue component (0-255).
|
||||
* @param a Alpha component (0-255), defaults to 255 (fully opaque).
|
||||
*/
|
||||
constructor(r: Int, g: Int, b: Int, a: Int = 255) : this(String.format("#%02x%02x%02x%02x", r, g, b, a))
|
||||
|
||||
/**
|
||||
* The red component of the color (0-255).
|
||||
*/
|
||||
val r: Int = hexCode.substring(1..2).toInt(16)
|
||||
|
||||
/**
|
||||
* The green component of the color (0-255).
|
||||
*/
|
||||
val g: Int = hexCode.substring(3..4).toInt(16)
|
||||
|
||||
/**
|
||||
* The blue component of the color (0-255).
|
||||
*/
|
||||
val b: Int = hexCode.substring(5..6).toInt(16)
|
||||
|
||||
/**
|
||||
* The alpha component of the color (0-255).
|
||||
* Defaults to 255 (fully opaque) if not specified in the hex code.
|
||||
*/
|
||||
val a: Int = if (hexCode.length == 9) hexCode.substring(7..8).toInt(16) else 255
|
||||
|
||||
init {
|
||||
require(hexCode.matches(Regex("^#[0-9A-Fa-f]{6}([0-9A-Fa-f]{2})?$"))) {
|
||||
"Hex code should be in format #RRGGBB[AA]"
|
||||
}
|
||||
}
|
||||
}
|
@@ -1,31 +0,0 @@
|
||||
/*
|
||||
* Copyright 2003-2025 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.api.models
|
||||
|
||||
import org.jetbrains.annotations.Range
|
||||
|
||||
/**
|
||||
* Represents a Vim jump location.
|
||||
*/
|
||||
data class Jump(
|
||||
/**
|
||||
* The 0-based line number of the jump.
|
||||
*/
|
||||
val line: @Range(from = 0, to = Int.MAX_VALUE.toLong()) Int,
|
||||
|
||||
/**
|
||||
* The 0-based column number of the jump.
|
||||
*/
|
||||
val col: @Range(from = 0, to = Int.MAX_VALUE.toLong()) Int,
|
||||
|
||||
/**
|
||||
* The file path where the jump is located.
|
||||
*/
|
||||
val filepath: Path,
|
||||
)
|
@@ -1,36 +0,0 @@
|
||||
/*
|
||||
* Copyright 2003-2025 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.api.models
|
||||
|
||||
import org.jetbrains.annotations.Range
|
||||
|
||||
/**
|
||||
* Represents a Vim mark.
|
||||
*/
|
||||
data class Mark(
|
||||
/**
|
||||
* The character key of the mark (a-z for local marks, A-Z for global marks).
|
||||
*/
|
||||
val key: Char,
|
||||
|
||||
/**
|
||||
* The 0-based line number of the mark.
|
||||
*/
|
||||
val line: @Range(from = 0, to = Int.MAX_VALUE.toLong()) Int,
|
||||
|
||||
/**
|
||||
* The 0-based column number of the mark.
|
||||
*/
|
||||
val col: @Range(from = 0, to = Int.MAX_VALUE.toLong()) Int,
|
||||
|
||||
/**
|
||||
* The file path where the mark is located.
|
||||
*/
|
||||
val filePath: Path,
|
||||
)
|
@@ -1,140 +0,0 @@
|
||||
/*
|
||||
* Copyright 2003-2025 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.api.models
|
||||
|
||||
|
||||
/**
|
||||
* Represents a Vim editor mode.
|
||||
*/
|
||||
enum class Mode {
|
||||
/**
|
||||
* Normal mode - the default mode where commands and motions can be executed.
|
||||
*/
|
||||
NORMAL,
|
||||
|
||||
/**
|
||||
* Operator-pending mode - entered after an operator command is given but before a motion is provided.
|
||||
*/
|
||||
OP_PENDING,
|
||||
|
||||
/**
|
||||
* Operator-pending mode with forced characterwise operation.
|
||||
*/
|
||||
OP_PENDING_CHARACTERWISE,
|
||||
|
||||
/**
|
||||
* Operator-pending mode with forced linewise operation.
|
||||
*/
|
||||
OP_PENDING_LINEWISE,
|
||||
|
||||
/**
|
||||
* Operator-pending mode with forced blockwise operation.
|
||||
*/
|
||||
OP_PENDING_BLOCKWISE,
|
||||
|
||||
/**
|
||||
* Normal mode using i_CTRL-O in Insert-mode.
|
||||
*/
|
||||
NORMAL_FROM_INSERT,
|
||||
|
||||
/**
|
||||
* Normal mode using i_CTRL-O in Replace-mode.
|
||||
*/
|
||||
NORMAL_FROM_REPLACE,
|
||||
|
||||
/**
|
||||
* Normal mode using i_CTRL-O in Virtual-Replace-mode.
|
||||
*/
|
||||
NORMAL_FROM_VIRTUAL_REPLACE,
|
||||
|
||||
/**
|
||||
* Visual mode with character-wise selection.
|
||||
*/
|
||||
VISUAL_CHARACTER,
|
||||
|
||||
/**
|
||||
* Visual mode with character-wise selection using v_CTRL-O in Select mode.
|
||||
*/
|
||||
VISUAL_CHARACTER_FROM_SELECT,
|
||||
|
||||
/**
|
||||
* Visual mode with line-wise selection.
|
||||
*/
|
||||
VISUAL_LINE,
|
||||
|
||||
/**
|
||||
* Visual mode with line-wise selection using v_CTRL-O in Select mode.
|
||||
*/
|
||||
VISUAL_LINE_FROM_SELECT,
|
||||
|
||||
/**
|
||||
* Visual mode with block-wise selection.
|
||||
*/
|
||||
VISUAL_BLOCK,
|
||||
|
||||
/**
|
||||
* Visual mode with block-wise selection using v_CTRL-O in Select mode.
|
||||
*/
|
||||
VISUAL_BLOCK_FROM_SELECT,
|
||||
|
||||
/**
|
||||
* Select mode with character-wise selection.
|
||||
*/
|
||||
SELECT_CHARACTER,
|
||||
|
||||
/**
|
||||
* Select mode with line-wise selection.
|
||||
*/
|
||||
SELECT_LINE,
|
||||
|
||||
/**
|
||||
* Select mode with block-wise selection.
|
||||
*/
|
||||
SELECT_BLOCK,
|
||||
|
||||
/**
|
||||
* Insert mode - used for inserting text.
|
||||
*/
|
||||
INSERT,
|
||||
|
||||
/**
|
||||
* Replace mode - used for replacing existing text.
|
||||
*/
|
||||
REPLACE,
|
||||
|
||||
/**
|
||||
* Command-line mode - used for entering Ex commands.
|
||||
*/
|
||||
COMMAND_LINE;
|
||||
|
||||
/**
|
||||
* Returns the TextType associated with this mode, if applicable.
|
||||
* Only visual and select modes have a TextType.
|
||||
*/
|
||||
val selectionType: TextType?
|
||||
get() = when (this) {
|
||||
VISUAL_CHARACTER, VISUAL_CHARACTER_FROM_SELECT, SELECT_CHARACTER -> TextType.CHARACTER_WISE
|
||||
VISUAL_LINE, VISUAL_LINE_FROM_SELECT, SELECT_LINE -> TextType.LINE_WISE
|
||||
VISUAL_BLOCK, VISUAL_BLOCK_FROM_SELECT, SELECT_BLOCK -> TextType.BLOCK_WISE
|
||||
else -> null
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns true if this mode is a visual mode.
|
||||
*/
|
||||
val isVisual: Boolean
|
||||
get() = this == VISUAL_CHARACTER || this == VISUAL_LINE || this == VISUAL_BLOCK ||
|
||||
this == VISUAL_CHARACTER_FROM_SELECT || this == VISUAL_LINE_FROM_SELECT || this == VISUAL_BLOCK_FROM_SELECT
|
||||
|
||||
/**
|
||||
* Returns true if this mode is a select mode.
|
||||
*/
|
||||
val isSelect: Boolean
|
||||
get() = this == SELECT_CHARACTER || this == SELECT_LINE || this == SELECT_BLOCK
|
||||
}
|
@@ -1,26 +0,0 @@
|
||||
/*
|
||||
* Copyright 2003-2025 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.api.models
|
||||
|
||||
/**
|
||||
* Represents a path.
|
||||
*/
|
||||
interface Path {
|
||||
/**
|
||||
* The protocol part of the path.
|
||||
*/
|
||||
val protocol: String
|
||||
|
||||
/**
|
||||
* The segments of the path as an array of strings.
|
||||
*/
|
||||
val path: Array<String>
|
||||
|
||||
companion object
|
||||
}
|
@@ -1,42 +0,0 @@
|
||||
/*
|
||||
* Copyright 2003-2025 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.api.models
|
||||
|
||||
/**
|
||||
* Represents a range of text in the editor.
|
||||
* Can be either a simple linear range or a block (rectangular) range.
|
||||
*/
|
||||
sealed interface Range {
|
||||
/**
|
||||
* Represents a simple linear range of text from start to end offset.
|
||||
*
|
||||
* @property start The starting offset of the range.
|
||||
* @property end The ending offset of the range (exclusive).
|
||||
*/
|
||||
data class Simple(val start: Int, val end: Int) : Range
|
||||
|
||||
/**
|
||||
* Represents a block (rectangular) selection consisting of multiple simple ranges.
|
||||
* Each simple range typically represents a line segment in the block selection.
|
||||
*
|
||||
* @property ranges An array of simple ranges that make up the block selection.
|
||||
*/
|
||||
data class Block(val ranges: Array<Simple>) : Range {
|
||||
override fun equals(other: Any?): Boolean {
|
||||
if (this === other) return true
|
||||
if (javaClass != other?.javaClass) return false
|
||||
other as Block
|
||||
return ranges.contentEquals(other.ranges)
|
||||
}
|
||||
|
||||
override fun hashCode(): Int {
|
||||
return ranges.contentHashCode()
|
||||
}
|
||||
}
|
||||
}
|
@@ -1,39 +0,0 @@
|
||||
/*
|
||||
* Copyright 2003-2025 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.api.scopes
|
||||
|
||||
/**
|
||||
* Scope for functions that provide working with digraphs.
|
||||
*/
|
||||
@VimApiDsl
|
||||
interface DigraphScope {
|
||||
/**
|
||||
* Gets the character for a digraph.
|
||||
*
|
||||
* In Vim, this is equivalent to entering CTRL-K followed by the two characters in insert mode.
|
||||
* Example: CTRL-K a: produces 'ä'
|
||||
*
|
||||
* @param ch1 The first character of the digraph
|
||||
* @param ch2 The second character of the digraph
|
||||
* @return The Unicode codepoint of the character represented by the digraph, or the codepoint of ch2 if no digraph is found
|
||||
*/
|
||||
fun getCharacter(ch1: Char, ch2: Char): Int
|
||||
|
||||
/**
|
||||
* Adds a custom digraph.
|
||||
*
|
||||
* In Vim, this is equivalent to the `:digraph` command with arguments.
|
||||
* Example: `:digraph a: 228` adds a digraph where 'a:' produces 'ä'
|
||||
*
|
||||
* @param ch1 The first character of the digraph
|
||||
* @param ch2 The second character of the digraph
|
||||
* @param codepoint The Unicode codepoint of the character to associate with the digraph
|
||||
*/
|
||||
fun add(ch1: Char, ch2: Char, codepoint: Int)
|
||||
}
|
@@ -1,189 +0,0 @@
|
||||
/*
|
||||
* Copyright 2003-2025 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.api.scopes
|
||||
|
||||
import com.intellij.vim.api.VimApi
|
||||
import com.intellij.vim.api.models.CaretId
|
||||
import com.intellij.vim.api.models.Mode
|
||||
import com.intellij.vim.api.models.Range
|
||||
|
||||
/**
|
||||
* Scope that provides access to various listeners.
|
||||
*/
|
||||
@VimApiDsl
|
||||
interface ListenersScope {
|
||||
/**
|
||||
* Registers a callback that is invoked when the editor mode changes.
|
||||
*
|
||||
* The callback receives the previous mode as a parameter.
|
||||
*
|
||||
* Example:
|
||||
* ```kotlin
|
||||
* listeners {
|
||||
* onModeChange { oldMode ->
|
||||
* if (mode == Mode.INSERT) {
|
||||
* // Do something when entering INSERT mode
|
||||
* }
|
||||
* }
|
||||
* }
|
||||
* ```
|
||||
*
|
||||
* @param callback The function to execute when the mode changes
|
||||
*/
|
||||
fun onModeChange(callback: suspend VimApi.(Mode) -> Unit)
|
||||
|
||||
/**
|
||||
* Registers a callback that is invoked when text is yanked.
|
||||
*
|
||||
* The callback receives a map of caret IDs to the yanked text ranges.
|
||||
*
|
||||
* Example:
|
||||
* ```kotlin
|
||||
* listeners {
|
||||
* onYank { caretRangeMap ->
|
||||
* // Process yanked text ranges
|
||||
* caretRangeMap.forEach { (caretId, range) ->
|
||||
* // Highlight or process the yanked range
|
||||
* }
|
||||
* }
|
||||
* }
|
||||
* ```
|
||||
*
|
||||
* @param callback The function to execute when text is yanked
|
||||
*/
|
||||
fun onYank(callback: suspend VimApi.(Map<CaretId, Range.Simple>) -> Unit)
|
||||
|
||||
/**
|
||||
* Registers a callback that is invoked when a new editor is created.
|
||||
*
|
||||
* Example:
|
||||
* ```kotlin
|
||||
* listeners {
|
||||
* onEditorCreate {
|
||||
* // Initialize resources for the new editor
|
||||
* }
|
||||
* }
|
||||
* ```
|
||||
*
|
||||
* @param callback The function to execute when an editor is created
|
||||
*/
|
||||
fun onEditorCreate(callback: suspend VimApi.() -> Unit)
|
||||
|
||||
/**
|
||||
* Registers a callback that is invoked when an editor is released.
|
||||
*
|
||||
* Example:
|
||||
* ```kotlin
|
||||
* listeners {
|
||||
* onEditorRelease {
|
||||
* // Clean up resources associated with the editor
|
||||
* }
|
||||
* }
|
||||
* ```
|
||||
*
|
||||
* @param callback The function to execute when an editor is released
|
||||
*/
|
||||
fun onEditorRelease(callback: suspend VimApi.() -> Unit)
|
||||
|
||||
/**
|
||||
* Registers a callback that is invoked when an editor gains focus.
|
||||
*
|
||||
* Example:
|
||||
* ```kotlin
|
||||
* listeners {
|
||||
* onEditorFocusGain {
|
||||
* // Perform actions when editor gains focus
|
||||
* }
|
||||
* }
|
||||
* ```
|
||||
*
|
||||
* @param callback The function to execute when an editor gains focus
|
||||
*/
|
||||
fun onEditorFocusGain(callback: suspend VimApi.() -> Unit)
|
||||
|
||||
/**
|
||||
* Registers a callback that is invoked when an editor loses focus.
|
||||
*
|
||||
* Example:
|
||||
* ```kotlin
|
||||
* listeners {
|
||||
* onEditorFocusLost {
|
||||
* // Perform actions when editor loses focus
|
||||
* }
|
||||
* }
|
||||
* ```
|
||||
*
|
||||
* @param callback The function to execute when an editor loses focus
|
||||
*/
|
||||
fun onEditorFocusLost(callback: suspend VimApi.() -> Unit)
|
||||
|
||||
/**
|
||||
* Registers a callback that is invoked when macro recording starts.
|
||||
*
|
||||
* Example:
|
||||
* ```kotlin
|
||||
* listeners {
|
||||
* onMacroRecordingStart {
|
||||
* // Perform actions when macro recording begins
|
||||
* }
|
||||
* }
|
||||
* ```
|
||||
*
|
||||
* @param callback The function to execute when macro recording starts
|
||||
*/
|
||||
fun onMacroRecordingStart(callback: suspend VimApi.() -> Unit)
|
||||
|
||||
/**
|
||||
* Registers a callback that is invoked when macro recording finishes.
|
||||
*
|
||||
* Example:
|
||||
* ```kotlin
|
||||
* listeners {
|
||||
* onMacroRecordingFinish {
|
||||
* // Perform actions when macro recording ends
|
||||
* }
|
||||
* }
|
||||
* ```
|
||||
*
|
||||
* @param callback The function to execute when macro recording finishes
|
||||
*/
|
||||
fun onMacroRecordingFinish(callback: suspend VimApi.() -> Unit)
|
||||
|
||||
/**
|
||||
* Registers a callback that is invoked when IdeaVim is enabled.
|
||||
*
|
||||
* Example usage:
|
||||
* ```kotlin
|
||||
* listeners {
|
||||
* onIdeaVimEnabled {
|
||||
* // Initialize plugin resources when IdeaVim is enabled
|
||||
* }
|
||||
* }
|
||||
* ```
|
||||
*
|
||||
* @param callback The function to execute when IdeaVim is enabled
|
||||
*/
|
||||
fun onIdeaVimEnabled(callback: suspend VimApi.() -> Unit)
|
||||
|
||||
/**
|
||||
* Registers a callback that is invoked when IdeaVim is disabled.
|
||||
*
|
||||
* Example usage:
|
||||
* ```kotlin
|
||||
* listeners {
|
||||
* onIdeaVimDisabled {
|
||||
* // Clean up plugin resources when IdeaVim is disabled
|
||||
* }
|
||||
* }
|
||||
* ```
|
||||
*
|
||||
* @param callback The function to execute when IdeaVim is disabled
|
||||
*/
|
||||
fun onIdeaVimDisabled(callback: suspend VimApi.() -> Unit)
|
||||
}
|
@@ -1,516 +0,0 @@
|
||||
/*
|
||||
* Copyright 2003-2025 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.api.scopes
|
||||
|
||||
import com.intellij.vim.api.VimApi
|
||||
|
||||
/**
|
||||
* Scope that provides access to mappings.
|
||||
*/
|
||||
@VimApiDsl
|
||||
interface MappingScope {
|
||||
/**
|
||||
* Maps a [from] key sequence to [to] in normal mode.
|
||||
*/
|
||||
fun nmap(from: String, to: String)
|
||||
|
||||
/**
|
||||
* Removes a [keys] mapping in normal mode.
|
||||
*
|
||||
* The [keys] must fully match the 'from' keys of the original mapping.
|
||||
*
|
||||
* Example:
|
||||
* ```kotlin
|
||||
* nmap("abc", "def") // Create mapping
|
||||
* nunmap("a") // × Does not unmap anything
|
||||
* nunmap("abc") // ✓ Properly unmaps the mapping
|
||||
* ```
|
||||
*/
|
||||
fun nunmap(keys: String)
|
||||
|
||||
/**
|
||||
* Maps a [from] key sequence to [to] in visual mode.
|
||||
*/
|
||||
fun vmap(from: String, to: String)
|
||||
|
||||
/**
|
||||
* Removes a [keys] mapping in visual mode.
|
||||
*
|
||||
* The [keys] must fully match the 'from' keys of the original mapping.
|
||||
*
|
||||
* Example:
|
||||
* ```kotlin
|
||||
* vmap("abc", "def") // Create mapping
|
||||
* vunmap("a") // × Does not unmap anything
|
||||
* vunmap("abc") // ✓ Properly unmaps the mapping
|
||||
* ```
|
||||
*/
|
||||
fun vunmap(keys: String)
|
||||
|
||||
/**
|
||||
* Maps a [from] key sequence to an [action] in normal mode.
|
||||
*/
|
||||
fun nmap(from: String, action: suspend VimApi.() -> Unit)
|
||||
|
||||
/**
|
||||
* Maps a [from] key sequence to an [action] in visual mode.
|
||||
*/
|
||||
fun vmap(from: String, action: suspend VimApi.() -> Unit)
|
||||
|
||||
/**
|
||||
* Maps [keys] to an [action] with an [actionName] in normal mode.
|
||||
*
|
||||
* [actionName] is needed to provide an intermediate mapping from the [keys] to [action].
|
||||
* Two mappings will be created: from [keys] to [actionName] and from [actionName] to [action].
|
||||
* In this way, the user will be able to rewrite the default mapping to the plugin by
|
||||
* providing a custom mapping to [actionName].
|
||||
*/
|
||||
fun nmap(
|
||||
keys: String,
|
||||
actionName: String,
|
||||
action: suspend VimApi.() -> Unit,
|
||||
)
|
||||
|
||||
/**
|
||||
* Maps [keys] to an [action] with an [actionName] in visual mode.
|
||||
*
|
||||
* [actionName] is needed to provide an intermediate mapping from the [keys] to [action].
|
||||
* Two mappings will be created: from [keys] to [actionName] and from [actionName] to [action].
|
||||
* In this way, the user will be able to rewrite the default mapping to the plugin by
|
||||
* providing a custom mapping to [actionName].
|
||||
*/
|
||||
fun vmap(
|
||||
keys: String,
|
||||
actionName: String,
|
||||
action: suspend VimApi.() -> Unit,
|
||||
)
|
||||
|
||||
/**
|
||||
* Maps a [from] key sequence to [to] in all modes.
|
||||
*/
|
||||
fun map(from: String, to: String)
|
||||
|
||||
/**
|
||||
* Removes a [keys] mapping in all modes.
|
||||
*
|
||||
* The [keys] must fully match the 'from' keys of the original mapping.
|
||||
*
|
||||
* Example:
|
||||
* ```kotlin
|
||||
* map("abc", "def") // Create mapping
|
||||
* unmap("a") // × Does not unmap anything
|
||||
* unmap("abc") // ✓ Properly unmaps the mapping
|
||||
* ```
|
||||
*/
|
||||
fun unmap(keys: String)
|
||||
|
||||
/**
|
||||
* Maps a [from] key sequence to an [action] in all modes.
|
||||
*/
|
||||
fun map(from: String, action: suspend VimApi.() -> Unit)
|
||||
|
||||
/**
|
||||
* Maps [keys] to an [action] with an [actionName] in all modes.
|
||||
*
|
||||
* [actionName] is needed to provide an intermediate mapping from the [keys] to [action].
|
||||
* Two mappings will be created: from [keys] to [actionName] and from [actionName] to [action].
|
||||
* In this way, the user will be able to rewrite the default mapping to the plugin by
|
||||
* providing a custom mapping to [actionName].
|
||||
*/
|
||||
fun map(
|
||||
keys: String,
|
||||
actionName: String,
|
||||
action: suspend VimApi.() -> Unit,
|
||||
)
|
||||
|
||||
/**
|
||||
* Maps a [from] key sequence to [to] in visual exclusive mode.
|
||||
*/
|
||||
fun xmap(from: String, to: String)
|
||||
|
||||
/**
|
||||
* Removes a [keys] mapping in visual exclusive mode.
|
||||
*
|
||||
* The [keys] must fully match the 'from' keys of the original mapping.
|
||||
*
|
||||
* Example:
|
||||
* ```kotlin
|
||||
* xmap("abc", "def") // Create mapping
|
||||
* xunmap("a") // × Does not unmap anything
|
||||
* xunmap("abc") // ✓ Properly unmaps the mapping
|
||||
* ```
|
||||
*/
|
||||
fun xunmap(keys: String)
|
||||
|
||||
/**
|
||||
* Maps a [from] key sequence to an [action] in visual exclusive mode.
|
||||
*/
|
||||
fun xmap(from: String, action: suspend VimApi.() -> Unit)
|
||||
|
||||
/**
|
||||
* Maps [keys] to an [action] with an [actionName] in visual exclusive mode.
|
||||
*
|
||||
* [actionName] is needed to provide an intermediate mapping from the [keys] to [action].
|
||||
* Two mappings will be created: from [keys] to [actionName] and from [actionName] to [action].
|
||||
* In this way, the user will be able to rewrite the default mapping to the plugin by
|
||||
* providing a custom mapping to [actionName].
|
||||
*/
|
||||
fun xmap(
|
||||
keys: String,
|
||||
actionName: String,
|
||||
action: suspend VimApi.() -> Unit,
|
||||
)
|
||||
|
||||
/**
|
||||
* Maps a [from] key sequence to [to] in select mode.
|
||||
*/
|
||||
fun smap(from: String, to: String)
|
||||
|
||||
/**
|
||||
* Removes a [keys] mapping in select mode.
|
||||
*
|
||||
* The [keys] must fully match the 'from' keys of the original mapping.
|
||||
*
|
||||
* Example:
|
||||
* ```kotlin
|
||||
* smap("abc", "def") // Create mapping
|
||||
* sunmap("a") // × Does not unmap anything
|
||||
* sunmap("abc") // ✓ Properly unmaps the mapping
|
||||
* ```
|
||||
*/
|
||||
fun sunmap(keys: String)
|
||||
|
||||
/**
|
||||
* Maps a [from] key sequence to an [action] in select mode.
|
||||
*/
|
||||
fun smap(from: String, action: suspend VimApi.() -> Unit)
|
||||
|
||||
/**
|
||||
* Maps [keys] to an [action] with an [actionName] in select mode.
|
||||
*
|
||||
* [actionName] is needed to provide an intermediate mapping from the [keys] to [action].
|
||||
* Two mappings will be created: from [keys] to [actionName] and from [actionName] to [action].
|
||||
* In this way, the user will be able to rewrite the default mapping to the plugin by
|
||||
* providing a custom mapping to [actionName].
|
||||
*/
|
||||
fun smap(
|
||||
keys: String,
|
||||
actionName: String,
|
||||
action: suspend VimApi.() -> Unit,
|
||||
)
|
||||
|
||||
/**
|
||||
* Maps a [from] key sequence to [to] in operator pending mode.
|
||||
*/
|
||||
fun omap(from: String, to: String)
|
||||
|
||||
/**
|
||||
* Removes a [keys] mapping in operator pending mode.
|
||||
*
|
||||
* The [keys] must fully match the 'from' keys of the original mapping.
|
||||
*
|
||||
* Example:
|
||||
* ```kotlin
|
||||
* omap("abc", "def") // Create mapping
|
||||
* ounmap("a") // × Does not unmap anything
|
||||
* ounmap("abc") // ✓ Properly unmaps the mapping
|
||||
* ```
|
||||
*/
|
||||
fun ounmap(keys: String)
|
||||
|
||||
/**
|
||||
* Maps a [from] key sequence to an [action] in operator pending mode.
|
||||
*/
|
||||
fun omap(from: String, action: suspend VimApi.() -> Unit)
|
||||
|
||||
/**
|
||||
* Maps [keys] to an [action] with an [actionName] in operator pending mode.
|
||||
*
|
||||
* [actionName] is needed to provide an intermediate mapping from the [keys] to [action].
|
||||
* Two mappings will be created: from [keys] to [actionName] and from [actionName] to [action].
|
||||
* In this way, the user will be able to rewrite the default mapping to the plugin by
|
||||
* providing a custom mapping to [actionName].
|
||||
*/
|
||||
fun omap(
|
||||
keys: String,
|
||||
actionName: String,
|
||||
action: suspend VimApi.() -> Unit,
|
||||
)
|
||||
|
||||
/**
|
||||
* Maps a [from] key sequence to [to] in insert mode.
|
||||
*/
|
||||
fun imap(from: String, to: String)
|
||||
|
||||
/**
|
||||
* Removes a [keys] mapping in insert mode.
|
||||
*
|
||||
* The [keys] must fully match the 'from' keys of the original mapping.
|
||||
*
|
||||
* Example:
|
||||
* ```kotlin
|
||||
* imap("abc", "def") // Create mapping
|
||||
* iunmap("a") // × Does not unmap anything
|
||||
* iunmap("abc") // ✓ Properly unmaps the mapping
|
||||
* ```
|
||||
*/
|
||||
fun iunmap(keys: String)
|
||||
|
||||
/**
|
||||
* Maps a [from] key sequence to an [action] in insert mode.
|
||||
*/
|
||||
fun imap(from: String, action: suspend VimApi.() -> Unit)
|
||||
|
||||
/**
|
||||
* Maps [keys] to an [action] with an [actionName] in insert mode.
|
||||
*
|
||||
* [actionName] is needed to provide an intermediate mapping from the [keys] to [action].
|
||||
* Two mappings will be created: from [keys] to [actionName] and from [actionName] to [action].
|
||||
* In this way, the user will be able to rewrite the default mapping to the plugin by
|
||||
* providing a custom mapping to [actionName].
|
||||
*/
|
||||
fun imap(
|
||||
keys: String,
|
||||
actionName: String,
|
||||
action: suspend VimApi.() -> Unit,
|
||||
)
|
||||
|
||||
/**
|
||||
* Maps a [from] key sequence to [to] in command line mode.
|
||||
*/
|
||||
fun cmap(from: String, to: String)
|
||||
|
||||
/**
|
||||
* Removes a [keys] mapping in command line mode.
|
||||
*
|
||||
* The [keys] must fully match the 'from' keys of the original mapping.
|
||||
*
|
||||
* Example:
|
||||
* ```kotlin
|
||||
* cmap("abc", "def") // Create mapping
|
||||
* cunmap("a") // × Does not unmap anything
|
||||
* cunmap("abc") // ✓ Properly unmaps the mapping
|
||||
* ```
|
||||
*/
|
||||
fun cunmap(keys: String)
|
||||
|
||||
/**
|
||||
* Maps a [from] key sequence to an [action] in command line mode.
|
||||
*/
|
||||
fun cmap(from: String, action: suspend VimApi.() -> Unit)
|
||||
|
||||
/**
|
||||
* Maps [keys] to an [action] with an [actionName] in command line mode.
|
||||
*
|
||||
* [actionName] is needed to provide an intermediate mapping from the [keys] to [action].
|
||||
* Two mappings will be created: from [keys] to [actionName] and from [actionName] to [action].
|
||||
* In this way, the user will be able to rewrite the default mapping to the plugin by
|
||||
* providing a custom mapping to [actionName].
|
||||
*/
|
||||
fun cmap(
|
||||
keys: String,
|
||||
actionName: String,
|
||||
action: suspend VimApi.() -> Unit,
|
||||
)
|
||||
|
||||
/**
|
||||
* Maps a [from] key sequence to [to] in normal mode non-recursively.
|
||||
*/
|
||||
fun nnoremap(from: String, to: String)
|
||||
|
||||
/**
|
||||
* Maps a [from] key sequence to an [action] in normal mode non-recursively.
|
||||
*/
|
||||
fun nnoremap(from: String, action: suspend VimApi.() -> Unit)
|
||||
|
||||
/**
|
||||
* Maps [keys] to an [action] with an [actionName] in normal mode non-recursively.
|
||||
*
|
||||
* [actionName] is needed to provide an intermediate mapping from the [keys] to [action].
|
||||
* Two mappings will be created: from [keys] to [actionName] and from [actionName] to [action].
|
||||
* In this way, the user will be able to rewrite the default mapping to the plugin by
|
||||
* providing a custom mapping to [actionName].
|
||||
*/
|
||||
fun nnoremap(
|
||||
keys: String,
|
||||
actionName: String,
|
||||
action: suspend VimApi.() -> Unit,
|
||||
)
|
||||
|
||||
/**
|
||||
* Maps a [from] key sequence to [to] in visual mode non-recursively.
|
||||
*/
|
||||
fun vnoremap(from: String, to: String)
|
||||
|
||||
/**
|
||||
* Maps a [from] key sequence to an [action] in visual mode non-recursively.
|
||||
*/
|
||||
fun vnoremap(from: String, action: suspend VimApi.() -> Unit)
|
||||
|
||||
/**
|
||||
* Maps [keys] to an [action] with an [actionName] in visual mode non-recursively.
|
||||
*
|
||||
* [actionName] is needed to provide an intermediate mapping from the [keys] to [action].
|
||||
* Two mappings will be created: from [keys] to [actionName] and from [actionName] to [action].
|
||||
* In this way, the user will be able to rewrite the default mapping to the plugin by
|
||||
* providing a custom mapping to [actionName].
|
||||
*/
|
||||
fun vnoremap(
|
||||
keys: String,
|
||||
actionName: String,
|
||||
action: suspend VimApi.() -> Unit,
|
||||
)
|
||||
|
||||
/**
|
||||
* Maps a [from] key sequence to [to] in all modes non-recursively.
|
||||
*/
|
||||
fun noremap(from: String, to: String)
|
||||
|
||||
/**
|
||||
* Maps a [from] key sequence to an [action] in all modes non-recursively.
|
||||
*/
|
||||
fun noremap(from: String, action: suspend VimApi.() -> Unit)
|
||||
|
||||
/**
|
||||
* Maps [keys] to an [action] with an [actionName] in all modes non-recursively.
|
||||
*
|
||||
* [actionName] is needed to provide an intermediate mapping from the [keys] to [action].
|
||||
* Two mappings will be created: from [keys] to [actionName] and from [actionName] to [action].
|
||||
* In this way, the user will be able to rewrite the default mapping to the plugin by
|
||||
* providing a custom mapping to [actionName].
|
||||
*/
|
||||
fun noremap(
|
||||
keys: String,
|
||||
actionName: String,
|
||||
action: suspend VimApi.() -> Unit,
|
||||
)
|
||||
|
||||
/**
|
||||
* Maps a [from] key sequence to [to] in visual exclusive mode non-recursively.
|
||||
*/
|
||||
fun xnoremap(from: String, to: String)
|
||||
|
||||
/**
|
||||
* Maps a [from] key sequence to an [action] in visual exclusive mode non-recursively.
|
||||
*/
|
||||
fun xnoremap(from: String, action: suspend VimApi.() -> Unit)
|
||||
|
||||
/**
|
||||
* Maps [keys] to an [action] with an [actionName] in visual exclusive mode non-recursively.
|
||||
*
|
||||
* [actionName] is needed to provide an intermediate mapping from the [keys] to [action].
|
||||
* Two mappings will be created: from [keys] to [actionName] and from [actionName] to [action].
|
||||
* In this way, the user will be able to rewrite the default mapping to the plugin by
|
||||
* providing a custom mapping to [actionName].
|
||||
*/
|
||||
fun xnoremap(
|
||||
keys: String,
|
||||
actionName: String,
|
||||
action: suspend VimApi.() -> Unit,
|
||||
)
|
||||
|
||||
/**
|
||||
* Maps a [from] key sequence to [to] in select mode non-recursively.
|
||||
*/
|
||||
fun snoremap(from: String, to: String)
|
||||
|
||||
/**
|
||||
* Maps a [from] key sequence to an [action] in select mode non-recursively.
|
||||
*/
|
||||
fun snoremap(from: String, action: suspend VimApi.() -> Unit)
|
||||
|
||||
/**
|
||||
* Maps [keys] to an [action] with an [actionName] in select mode non-recursively.
|
||||
*
|
||||
* [actionName] is needed to provide an intermediate mapping from the [keys] to [action].
|
||||
* Two mappings will be created: from [keys] to [actionName] and from [actionName] to [action].
|
||||
* In this way, the user will be able to rewrite the default mapping to the plugin by
|
||||
* providing a custom mapping to [actionName].
|
||||
*/
|
||||
fun snoremap(
|
||||
keys: String,
|
||||
actionName: String,
|
||||
action: suspend VimApi.() -> Unit,
|
||||
)
|
||||
|
||||
/**
|
||||
* Maps a [from] key sequence to [to] in operator pending mode non-recursively.
|
||||
*/
|
||||
fun onoremap(from: String, to: String)
|
||||
|
||||
/**
|
||||
* Maps a [from] key sequence to an [action] in operator pending mode non-recursively.
|
||||
*/
|
||||
fun onoremap(from: String, action: suspend VimApi.() -> Unit)
|
||||
|
||||
/**
|
||||
* Maps [keys] to an [action] with an [actionName] in operator pending mode non-recursively.
|
||||
*
|
||||
* [actionName] is needed to provide an intermediate mapping from the [keys] to [action].
|
||||
* Two mappings will be created: from [keys] to [actionName] and from [actionName] to [action].
|
||||
* In this way, the user will be able to rewrite the default mapping to the plugin by
|
||||
* providing a custom mapping to [actionName].
|
||||
*/
|
||||
fun onoremap(
|
||||
keys: String,
|
||||
actionName: String,
|
||||
action: suspend VimApi.() -> Unit,
|
||||
)
|
||||
|
||||
/**
|
||||
* Maps a [from] key sequence to [to] in insert mode non-recursively.
|
||||
*/
|
||||
fun inoremap(from: String, to: String)
|
||||
|
||||
/**
|
||||
* Maps a [from] key sequence to an [action] in insert mode non-recursively.
|
||||
*/
|
||||
fun inoremap(from: String, action: suspend VimApi.() -> Unit)
|
||||
|
||||
/**
|
||||
* Maps [keys] to an [action] with an [actionName] in insert mode non-recursively.
|
||||
*
|
||||
* [actionName] is needed to provide an intermediate mapping from the [keys] to [action].
|
||||
* Two mappings will be created: from [keys] to [actionName] and from [actionName] to [action].
|
||||
* In this way, the user will be able to rewrite the default mapping to the plugin by
|
||||
* providing a custom mapping to [actionName].
|
||||
*/
|
||||
fun inoremap(
|
||||
keys: String,
|
||||
actionName: String,
|
||||
action: suspend VimApi.() -> Unit,
|
||||
)
|
||||
|
||||
/**
|
||||
* Maps a [from] key sequence to [to] in command line mode non-recursively.
|
||||
*/
|
||||
fun cnoremap(from: String, to: String)
|
||||
|
||||
/**
|
||||
* Maps a key sequence in command line mode to an action non-recursively.
|
||||
*
|
||||
* @param from The key sequence to map from
|
||||
* @param action The action to execute when the key sequence is pressed
|
||||
*/
|
||||
fun cnoremap(from: String, action: suspend VimApi.() -> Unit)
|
||||
|
||||
/**
|
||||
* Maps [keys] to an [action] with an [actionName] in command line mode non-recursively.
|
||||
*
|
||||
* [actionName] is needed to provide an intermediate mapping from the [keys] to [action].
|
||||
* Two mappings will be created: from [keys] to [actionName] and from [actionName] to [action].
|
||||
* In this way, the user will be able to rewrite the default mapping to the plugin by
|
||||
* providing a custom mapping to [actionName].
|
||||
*/
|
||||
fun cnoremap(
|
||||
keys: String,
|
||||
actionName: String,
|
||||
action: suspend VimApi.() -> Unit,
|
||||
)
|
||||
}
|
@@ -1,171 +0,0 @@
|
||||
/*
|
||||
* Copyright 2003-2025 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.api.scopes
|
||||
|
||||
import com.intellij.vim.api.VimApi
|
||||
|
||||
/**
|
||||
* Scope for working with modal input in IdeaVim.
|
||||
*
|
||||
* This scope provides methods for creating and managing modal input dialogs,
|
||||
* which can be used to get user input in a Vim-like way.
|
||||
*
|
||||
* The ModalInput interface supports:
|
||||
* - Single character input with [inputChar]
|
||||
* - String input with [inputString]
|
||||
* - Repeating input operations with [repeat] and [repeatWhile]
|
||||
* - Updating the input prompt with [updateLabel]
|
||||
* - Closing the current input dialog with [closeCurrentInput]
|
||||
*/
|
||||
@VimApiDsl
|
||||
interface ModalInput {
|
||||
/**
|
||||
* Updates the label of the modal input dialog during input processing.
|
||||
*
|
||||
* This method allows you to dynamically modify the label shown to the user based on the current state.
|
||||
*
|
||||
* Example usage:
|
||||
* ```kotlin
|
||||
* modalInput()
|
||||
* .updateLabel { currentLabel ->
|
||||
* "$currentLabel - Updated"
|
||||
* }
|
||||
* .inputChar("Enter character:") { char ->
|
||||
* // Process the character
|
||||
* }
|
||||
* ```
|
||||
*
|
||||
* @param block A function that takes the current label and returns a new label
|
||||
* @return This ModalInput instance for method chaining
|
||||
*/
|
||||
fun updateLabel(block: (String) -> String): ModalInput
|
||||
|
||||
/**
|
||||
* Repeats the input operation as long as the specified condition is true.
|
||||
*
|
||||
* This method allows you to collect multiple inputs from the user until a certain condition is met.
|
||||
* The condition is evaluated before each input operation.
|
||||
*
|
||||
* Example usage:
|
||||
* ```kotlin
|
||||
* var inputCount = 0
|
||||
*
|
||||
* modalInput()
|
||||
* .repeatWhile {
|
||||
* inputCount < 3 // Continue until we've received 3 inputs
|
||||
* }
|
||||
* .inputChar("Enter character:") { char ->
|
||||
* inputCount++
|
||||
* // Process the character
|
||||
* }
|
||||
* ```
|
||||
*
|
||||
* @param condition A function that returns true if the input operation should be repeated
|
||||
* @return This ModalInput instance for method chaining
|
||||
*/
|
||||
fun repeatWhile(condition: () -> Boolean): ModalInput
|
||||
|
||||
/**
|
||||
* Repeats the input operation a specified number of times.
|
||||
*
|
||||
* This method allows you to collect a fixed number of inputs from the user.
|
||||
*
|
||||
* Example usage:
|
||||
* ```kotlin
|
||||
* modalInput()
|
||||
* .repeat(3) // Get 3 characters from the user
|
||||
* .inputChar("Enter character:") { char ->
|
||||
* // Process each character as it's entered
|
||||
* // This handler will be called 3 times
|
||||
* }
|
||||
* ```
|
||||
*
|
||||
* @param count The number of times to repeat the input operation
|
||||
* @return This ModalInput instance for method chaining
|
||||
*/
|
||||
fun repeat(count: Int): ModalInput
|
||||
|
||||
/**
|
||||
* Creates a modal input dialog for collecting a string from the user.
|
||||
*
|
||||
* This method displays a dialog with the specified label and waits for the user to enter text.
|
||||
* The handler is executed after the user presses ENTER, receiving the entered string as a parameter.
|
||||
*
|
||||
* Example usage:
|
||||
* ```kotlin
|
||||
* modalInput()
|
||||
* .inputString("Enter string:") { enteredString ->
|
||||
* // Process the entered string
|
||||
* println("User entered: $enteredString")
|
||||
* }
|
||||
* ```
|
||||
*
|
||||
* This can be combined with other methods:
|
||||
*
|
||||
* ```kotlin
|
||||
* vimApi.modalInput()
|
||||
* .repeat(2) // Get two strings from the user
|
||||
* .inputString("Enter value:") { value ->
|
||||
* // Process each string as it's entered
|
||||
* }
|
||||
* ```
|
||||
*
|
||||
* @param label The label to display in the dialog
|
||||
* @param handler A function that will be called when the user enters input and presses ENTER
|
||||
*/
|
||||
fun inputString(label: String, handler: VimApi.(String) -> Unit)
|
||||
|
||||
/**
|
||||
* Creates a modal input dialog for collecting a single character from the user.
|
||||
*
|
||||
* This method displays a dialog with the specified label and waits for the user to press a key.
|
||||
* The handler is executed immediately after the user presses any key, receiving the entered character as a parameter.
|
||||
* Unlike [inputString], this method doesn't require the user to press ENTER.
|
||||
*
|
||||
* Example usage:
|
||||
* ```kotlin
|
||||
* vimApi.modalInput()
|
||||
* .inputChar("Press a key:") { char ->
|
||||
* // Process the entered character
|
||||
* when(char) {
|
||||
* 'y', 'Y' -> println("You confirmed")
|
||||
* 'n', 'N' -> println("You declined")
|
||||
* else -> println("Invalid option")
|
||||
* }
|
||||
* }
|
||||
* ```
|
||||
*
|
||||
* This can be combined with other methods:
|
||||
*
|
||||
* ```kotlin
|
||||
* vimApi.modalInput()
|
||||
* .repeatWhile { /* condition */ }
|
||||
* .inputChar("Enter character:") { char ->
|
||||
* // Process each character as it's entered
|
||||
* }
|
||||
* ```
|
||||
*
|
||||
* @param label The label to display in the dialog
|
||||
* @param handler A function that will be called when the user enters a character
|
||||
*/
|
||||
fun inputChar(label: String, handler: VimApi.(Char) -> Unit)
|
||||
|
||||
/**
|
||||
* Closes the current modal input dialog, if one is active.
|
||||
*
|
||||
* Example usage:
|
||||
* ```kotlin
|
||||
* modalInput().closeCurrentInput(refocusEditor = false)
|
||||
* ```
|
||||
*
|
||||
* @param refocusEditor Whether to refocus the editor after closing the dialog (default: true)
|
||||
* @return True if a dialog was closed, false if there was no active dialog
|
||||
*/
|
||||
fun closeCurrentInput(refocusEditor: Boolean = true): Boolean
|
||||
}
|
@@ -1,244 +0,0 @@
|
||||
/*
|
||||
* Copyright 2003-2025 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.api.scopes
|
||||
|
||||
import kotlin.reflect.KType
|
||||
import kotlin.reflect.typeOf
|
||||
|
||||
/**
|
||||
* Scope that provides functions for working with options.
|
||||
*/
|
||||
@VimApiDsl
|
||||
interface OptionScope {
|
||||
/**
|
||||
* Gets the value of an option with the specified type.
|
||||
*
|
||||
* **Note:** Prefer using the extension function `get<T>(name)` instead of calling this directly,
|
||||
* as it provides better type safety and cleaner syntax through reified type parameters.
|
||||
*
|
||||
* Example of preferred usage:
|
||||
* ```kotlin
|
||||
* myVimApi.option {
|
||||
* val ignoreCase = get<Boolean>("ignorecase")
|
||||
* val history = get<Int>("history")
|
||||
* val clipboard = get<String>("clipboard")
|
||||
* }
|
||||
* ```
|
||||
*
|
||||
* @param name The name of the option
|
||||
* @param type The KType of the option value
|
||||
* @return The value of the option
|
||||
* @throws IllegalArgumentException if the type is wrong or the option doesn't exist
|
||||
*/
|
||||
fun <T> getOptionValue(name: String, type: KType): T
|
||||
|
||||
/**
|
||||
* Sets an option value with the specified scope.
|
||||
*
|
||||
* **Note:** Prefer using the extension functions `set<T>(name, value)`, `setGlobal<T>(name, value)`,
|
||||
* or `setLocal<T>(name, value)` instead of calling this directly, as they provide better type safety
|
||||
* and cleaner syntax through reified type parameters.
|
||||
*
|
||||
* Example of preferred usage:
|
||||
* ```kotlin
|
||||
* myVimApi.option {
|
||||
* set("ignorecase", true) // Effective scope
|
||||
* setGlobal("number", 42) // Global scope
|
||||
* setLocal("tabstop", 4) // Local scope
|
||||
* }
|
||||
* ```
|
||||
*
|
||||
* @param name The name of the option
|
||||
* @param value The value to set
|
||||
* @param type The KType of the option value
|
||||
* @param scope The scope to set the option in ("global", "local", or "effective")
|
||||
* @throws IllegalArgumentException if the option doesn't exist or the type is wrong
|
||||
*/
|
||||
fun <T> setOption(name: String, value: T, type: KType, scope: String)
|
||||
|
||||
/**
|
||||
* Resets an option to its default value.
|
||||
*
|
||||
* In Vim, this is equivalent to `:set option&`.
|
||||
* Example: `:set ignorecase&` resets the 'ignorecase' option to its default value.
|
||||
*
|
||||
* @param name The name of the option
|
||||
*
|
||||
* @throws IllegalArgumentException if the option doesn't exist
|
||||
*/
|
||||
fun reset(name: String)
|
||||
|
||||
/**
|
||||
* Extension function to split a comma-separated option value into a list.
|
||||
* This is useful for processing list options like virtualedit, whichwrap, etc.
|
||||
*
|
||||
* Example:
|
||||
* ```kotlin
|
||||
* myVimApi.option {
|
||||
* val values = get<String>("virtualedit")?.split() ?: emptyList()
|
||||
* // "block,all" → ["block", "all"]
|
||||
* // "" → [""]
|
||||
* // "all" → ["all"]
|
||||
* }
|
||||
* ```
|
||||
*/
|
||||
fun String.split(): List<String> = split(",")
|
||||
}
|
||||
|
||||
/**
|
||||
* Gets the value of an option with the specified type.
|
||||
*
|
||||
* In Vim, options can be accessed with the `&` prefix.
|
||||
* Example: `&ignorecase` returns the value of the 'ignorecase' option.
|
||||
*
|
||||
* @param name The name of the option
|
||||
* @return The value of the option
|
||||
* @throws IllegalArgumentException if the type is wrong or the option doesn't exist
|
||||
*/
|
||||
inline fun <reified T> OptionScope.get(name: String): T {
|
||||
val kType: KType = typeOf<T>()
|
||||
return getOptionValue(name, kType)
|
||||
}
|
||||
|
||||
/**
|
||||
* Sets the global value of an option with the specified type.
|
||||
*
|
||||
* In Vim, this is equivalent to `:setglobal option=value`.
|
||||
* Example: `:setglobal ignorecase` or `let &g:ignorecase = 1`
|
||||
*
|
||||
* @param name The name of the option
|
||||
* @param value The value to set
|
||||
*
|
||||
* @throws IllegalArgumentException if the option doesn't exist or the type is wrong
|
||||
*/
|
||||
inline fun <reified T> OptionScope.setGlobal(name: String, value: T) {
|
||||
val kType: KType = typeOf<T>()
|
||||
setOption(name, value, kType, "global")
|
||||
}
|
||||
|
||||
/**
|
||||
* Sets the local value of an option with the specified type.
|
||||
*
|
||||
* In Vim, this is equivalent to `:setlocal option=value`.
|
||||
* Example: `:setlocal ignorecase` or `let &l:ignorecase = 1`
|
||||
*
|
||||
* @param name The name of the option
|
||||
* @param value The value to set
|
||||
*
|
||||
* @throws IllegalArgumentException if the option doesn't exist or the type is wrong
|
||||
*/
|
||||
inline fun <reified T> OptionScope.setLocal(name: String, value: T) {
|
||||
val kType: KType = typeOf<T>()
|
||||
setOption(name, value, kType, "local")
|
||||
}
|
||||
|
||||
/**
|
||||
* Sets the effective value of an option with the specified type.
|
||||
*
|
||||
* In Vim, this is equivalent to `:set option=value`.
|
||||
* Example: `:set ignorecase` or `let &ignorecase = 1`
|
||||
*
|
||||
* @param name The name of the option
|
||||
* @param value The value to set
|
||||
*
|
||||
* @throws IllegalArgumentException if the option doesn't exist or the type is wrong
|
||||
*/
|
||||
inline fun <reified T> OptionScope.set(name: String, value: T) {
|
||||
val kType: KType = typeOf<T>()
|
||||
setOption(name, value, kType, "effective")
|
||||
}
|
||||
|
||||
/**
|
||||
* Toggles a boolean option value.
|
||||
*
|
||||
* Example:
|
||||
* ```kotlin
|
||||
* myVimApi.option {
|
||||
* toggle("ignorecase") // true → false, false → true
|
||||
* }
|
||||
* ```
|
||||
*
|
||||
* @param name The name of the boolean option to toggle
|
||||
*/
|
||||
fun OptionScope.toggle(name: String) {
|
||||
val current = get<Boolean>(name)
|
||||
set(name, !current)
|
||||
}
|
||||
|
||||
/**
|
||||
* Appends values to a comma-separated list option.
|
||||
* This is equivalent to Vim's += operator for string options.
|
||||
* Duplicate values are not added.
|
||||
*
|
||||
* Example:
|
||||
* ```kotlin
|
||||
* myVimApi.option {
|
||||
* append("virtualedit", "block") // "" → "block"
|
||||
* append("virtualedit", "onemore") // "block" → "block,onemore"
|
||||
* append("virtualedit", "block") // "block,onemore" → "block,onemore" (no change)
|
||||
* }
|
||||
* ```
|
||||
*
|
||||
* @param name The name of the list option
|
||||
* @param values The values to append (duplicates will be ignored)
|
||||
*/
|
||||
fun OptionScope.append(name: String, vararg values: String) {
|
||||
val current = get<String>(name)
|
||||
val currentList = if (current.isEmpty()) emptyList() else current.split()
|
||||
val valuesToAdd = values.filterNot { it in currentList }
|
||||
val newList = currentList + valuesToAdd
|
||||
set(name, newList.joinToString(","))
|
||||
}
|
||||
|
||||
/**
|
||||
* Prepends values to a comma-separated list option.
|
||||
* This is equivalent to Vim's ^= operator for string options.
|
||||
* Duplicate values are not added.
|
||||
*
|
||||
* Example:
|
||||
* ```kotlin
|
||||
* myVimApi.option {
|
||||
* prepend("virtualedit", "block") // "all" → "block,all"
|
||||
* prepend("virtualedit", "onemore") // "block,all" → "onemore,block,all"
|
||||
* prepend("virtualedit", "all") // "onemore,block,all" → "onemore,block,all" (no change)
|
||||
* }
|
||||
* ```
|
||||
*
|
||||
* @param name The name of the list option
|
||||
* @param values The values to prepend (duplicates will be ignored)
|
||||
*/
|
||||
fun OptionScope.prepend(name: String, vararg values: String) {
|
||||
val current = get<String>(name)
|
||||
val currentList = if (current.isEmpty()) emptyList() else current.split()
|
||||
val valuesToAdd = values.filterNot { it in currentList }
|
||||
val newList = valuesToAdd + currentList
|
||||
set(name, newList.joinToString(","))
|
||||
}
|
||||
|
||||
/**
|
||||
* Removes values from a comma-separated list option.
|
||||
* This is equivalent to Vim's -= operator for string options.
|
||||
*
|
||||
* Example:
|
||||
* ```kotlin
|
||||
* myVimApi.option {
|
||||
* remove("virtualedit", "block") // "block,all" → "all"
|
||||
* remove("virtualedit", "all") // "all" → ""
|
||||
* }
|
||||
* ```
|
||||
*
|
||||
* @param name The name of the list option
|
||||
* @param values The values to remove
|
||||
*/
|
||||
fun OptionScope.remove(name: String, vararg values: String) {
|
||||
val current = get<String>(name)
|
||||
val currentList = if (current.isEmpty()) emptyList() else current.split()
|
||||
val newList = currentList.filterNot { it in values }
|
||||
set(name, newList.joinToString(","))
|
||||
}
|
@@ -1,60 +0,0 @@
|
||||
/*
|
||||
* Copyright 2003-2025 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.api.scopes
|
||||
|
||||
/**
|
||||
* Scope that provides functions for interacting with the Vim output panel.
|
||||
*/
|
||||
@VimApiDsl
|
||||
interface OutputPanelScope {
|
||||
/**
|
||||
* The text displayed in the output panel.
|
||||
*/
|
||||
val text: String
|
||||
|
||||
/**
|
||||
* The label text displayed at the bottom of the output panel.
|
||||
*
|
||||
* This is used for status information like "-- MORE --" to indicate
|
||||
* that there is more content to scroll through.
|
||||
*/
|
||||
val label: String
|
||||
|
||||
/**
|
||||
* Sets the text content of the output panel.
|
||||
*
|
||||
* This replaces any existing text in the panel with the provided text.
|
||||
*
|
||||
* @param text The new text to display in the output panel.
|
||||
*/
|
||||
fun setText(text: String)
|
||||
|
||||
/**
|
||||
* Appends text to the existing content of the output panel.
|
||||
*
|
||||
* @param text The text to append to the current content.
|
||||
* @param startNewLine Whether to start the appended text on a new line.
|
||||
* If true and there is an existing text, a newline character
|
||||
* will be inserted before the appended text.
|
||||
* Defaults to false.
|
||||
*/
|
||||
fun appendText(text: String, startNewLine: Boolean = false)
|
||||
|
||||
/**
|
||||
* Sets the label text at the bottom of the output panel.
|
||||
*
|
||||
* @param label The new label text to display.
|
||||
*/
|
||||
fun setLabel(label: String)
|
||||
|
||||
/**
|
||||
* Clears all text from the output panel.
|
||||
*/
|
||||
fun clearText()
|
||||
}
|
@@ -1,13 +0,0 @@
|
||||
/*
|
||||
* Copyright 2003-2025 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.api.scopes
|
||||
|
||||
@Target(AnnotationTarget.CLASS, AnnotationTarget.TYPE)
|
||||
@DslMarker
|
||||
internal annotation class VimApiDsl
|
@@ -1,32 +0,0 @@
|
||||
/*
|
||||
* Copyright 2003-2025 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.api.scopes.commandline
|
||||
|
||||
import com.intellij.vim.api.scopes.VimApiDsl
|
||||
|
||||
/**
|
||||
* Scope for command line functions that should be executed under read lock.
|
||||
*/
|
||||
@VimApiDsl
|
||||
interface CommandLineRead {
|
||||
/**
|
||||
* The text currently displayed in the command line.
|
||||
*/
|
||||
val text: String
|
||||
|
||||
/**
|
||||
* The current position of the caret in the command line.
|
||||
*/
|
||||
val caretPosition: Int
|
||||
|
||||
/**
|
||||
* True if the command line is currently active, false otherwise.
|
||||
*/
|
||||
val isActive: Boolean
|
||||
}
|
@@ -1,80 +0,0 @@
|
||||
/*
|
||||
* Copyright 2003-2025 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.api.scopes.commandline
|
||||
|
||||
import com.intellij.vim.api.VimApi
|
||||
import com.intellij.vim.api.scopes.VimApiDsl
|
||||
import kotlin.contracts.ExperimentalContracts
|
||||
import kotlin.contracts.InvocationKind
|
||||
import kotlin.contracts.contract
|
||||
|
||||
/**
|
||||
* Scope for interacting with the Vim command line.
|
||||
*/
|
||||
@VimApiDsl
|
||||
abstract class CommandLineScope {
|
||||
/**
|
||||
* Reads input from the command line and processes it with the provided function.
|
||||
*
|
||||
* @param prompt The prompt to display at the beginning of the command line.
|
||||
* @param finishOn The character that, when entered, will finish the input process. If null, only Enter will finish.
|
||||
* @param callback A function that will be called with the entered text when input is complete.
|
||||
*/
|
||||
abstract fun input(prompt: String, finishOn: Char? = null, callback: VimApi.(String) -> Unit)
|
||||
|
||||
/**
|
||||
* Executes operations on the command line that require a read lock.
|
||||
*
|
||||
* Example usage:
|
||||
* ```kotlin
|
||||
* commandLine {
|
||||
* read {
|
||||
* text
|
||||
* }
|
||||
* }
|
||||
* ```
|
||||
*
|
||||
* @param block A function with CommandLineRead receiver that contains the read operations to perform.
|
||||
* @return A Deferred that will complete with the result of the block execution.
|
||||
*/
|
||||
@OptIn(ExperimentalContracts::class)
|
||||
fun <T> read(block: CommandLineRead.() -> T): T {
|
||||
contract {
|
||||
callsInPlace(block, InvocationKind.EXACTLY_ONCE)
|
||||
}
|
||||
return this.ideRead(block)
|
||||
}
|
||||
|
||||
/**
|
||||
* Executes operations that require write lock on the command line.
|
||||
*
|
||||
* Example usage:
|
||||
* ```kotlin
|
||||
* // Set command line text
|
||||
* commandLineScope {
|
||||
* change {
|
||||
* setText("Hello")
|
||||
* }
|
||||
* }
|
||||
* ```
|
||||
*
|
||||
* @param block A function with CommandLineTransaction receiver that contains the write operations to perform.
|
||||
* @return A Job that represents the ongoing execution of the block.
|
||||
*/
|
||||
@OptIn(ExperimentalContracts::class)
|
||||
fun change(block: CommandLineTransaction.() -> Unit) {
|
||||
contract {
|
||||
callsInPlace(block, InvocationKind.EXACTLY_ONCE)
|
||||
}
|
||||
ideChange(block)
|
||||
}
|
||||
|
||||
protected abstract fun <T> ideRead(block: CommandLineRead.() -> T): T
|
||||
protected abstract fun ideChange(block: CommandLineTransaction.() -> Unit)
|
||||
}
|
@@ -1,47 +0,0 @@
|
||||
/*
|
||||
* Copyright 2003-2025 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.api.scopes.commandline
|
||||
|
||||
import com.intellij.vim.api.scopes.VimApiDsl
|
||||
|
||||
/**
|
||||
* Scope for command line functions that should be executed under write lock.
|
||||
*/
|
||||
@VimApiDsl
|
||||
interface CommandLineTransaction {
|
||||
/**
|
||||
* Sets the text content of the command line. It replaces any existing text in the command line with the provided text.
|
||||
*
|
||||
* @param text The new text to display in the command line.
|
||||
*/
|
||||
suspend fun setText(text: String)
|
||||
|
||||
/**
|
||||
* Inserts text at the specified position in the command line.
|
||||
*
|
||||
* @param offset The position at which to insert the text.
|
||||
* @param text The text to insert.
|
||||
*/
|
||||
suspend fun insertText(offset: Int, text: String)
|
||||
|
||||
/**
|
||||
* Sets the caret position in the command line.
|
||||
*
|
||||
* @param position The new position for the caret.
|
||||
*/
|
||||
suspend fun setCaretPosition(position: Int)
|
||||
|
||||
/**
|
||||
* Closes the command line.
|
||||
*
|
||||
* @param refocusEditor Whether to refocus the editor after closing the command line.
|
||||
* @return True if the command line was closed, false if it was not active.
|
||||
*/
|
||||
suspend fun close(refocusEditor: Boolean = true): Boolean
|
||||
}
|
@@ -1,328 +0,0 @@
|
||||
/*
|
||||
* Copyright 2003-2025 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.api.scopes.editor
|
||||
|
||||
import com.intellij.vim.api.models.CaretData
|
||||
import com.intellij.vim.api.models.CaretId
|
||||
import com.intellij.vim.api.models.Jump
|
||||
import com.intellij.vim.api.models.Line
|
||||
import com.intellij.vim.api.models.Mark
|
||||
import com.intellij.vim.api.models.Path
|
||||
import com.intellij.vim.api.models.Range
|
||||
import com.intellij.vim.api.scopes.VimApiDsl
|
||||
|
||||
/**
|
||||
* Interface giving functions to access the editor state. Does not imply read or write locks.
|
||||
*/
|
||||
@VimApiDsl
|
||||
interface EditorAccessor {
|
||||
/**
|
||||
* The total length of the text in the editor.
|
||||
*/
|
||||
val textLength: Long
|
||||
|
||||
/**
|
||||
* The entire text content of the editor.
|
||||
*/
|
||||
val text: CharSequence
|
||||
|
||||
/**
|
||||
* The total number of lines in the editor.
|
||||
*/
|
||||
val lineCount: Int
|
||||
|
||||
/**
|
||||
* File path of the editor.
|
||||
*/
|
||||
val filePath: Path
|
||||
|
||||
/**
|
||||
* Gets the start offset of the specified line.
|
||||
*
|
||||
* @param line The line number (0-based)
|
||||
* @return The offset of the first character in the line
|
||||
*/
|
||||
fun getLineStartOffset(line: Int): Int
|
||||
|
||||
/**
|
||||
* Gets the end offset of the specified line.
|
||||
*
|
||||
* @param line The line number (0-based)
|
||||
* @param allowEnd Whether to allow the end of the document as a valid result
|
||||
* @return The offset after the last character in the line
|
||||
*/
|
||||
fun getLineEndOffset(line: Int, allowEnd: Boolean): Int
|
||||
|
||||
/**
|
||||
* Gets information about the line containing the specified offset.
|
||||
*
|
||||
* @param offset The offset in the document
|
||||
* @return A Line object containing information about the line
|
||||
*/
|
||||
fun getLine(offset: Int): Line
|
||||
|
||||
/**
|
||||
* A list of data for all carets in the editor.
|
||||
*
|
||||
* Each element in the list is a CaretData object containing information about a caret,
|
||||
* such as its position, selection, and other properties.
|
||||
*/
|
||||
val caretData: List<CaretData>
|
||||
|
||||
/**
|
||||
* A list of IDs for all carets in the editor.
|
||||
*
|
||||
* These IDs can be used with the `with` function to perform operations on specific carets.
|
||||
*/
|
||||
val caretIds: List<CaretId>
|
||||
|
||||
/**
|
||||
* Gets a global mark by its character key.
|
||||
*
|
||||
* @param char The character key of the mark (A-Z)
|
||||
* @return The mark, or null if the mark doesn't exist
|
||||
*/
|
||||
fun getGlobalMark(char: Char): Mark?
|
||||
|
||||
/**
|
||||
* All global marks.
|
||||
*/
|
||||
val globalMarks: Set<Mark>
|
||||
|
||||
/**
|
||||
* Gets a jump from the jump list.
|
||||
*
|
||||
* @param count The number of jumps to go back (negative) or forward (positive) from the current position in the jump list.
|
||||
* @return The jump, or null if there is no jump at the specified position
|
||||
*/
|
||||
fun getJump(count: Int = 0): Jump?
|
||||
|
||||
/**
|
||||
* Gets all jumps in the jump list.
|
||||
*
|
||||
* @return A list of all jumps
|
||||
*/
|
||||
val jumps: List<Jump>
|
||||
|
||||
/**
|
||||
* Index of the current position in the jump list.
|
||||
*
|
||||
* This is used to determine which jump will be used when navigating with Ctrl-O and Ctrl-I.
|
||||
*/
|
||||
val currentJumpIndex: Int
|
||||
|
||||
/**
|
||||
* Scrolls the caret into view.
|
||||
*
|
||||
* This ensures that the caret is visible in the editor window.
|
||||
*/
|
||||
fun scrollCaretIntoView()
|
||||
|
||||
/**
|
||||
* Scrolls the editor by a specified number of lines.
|
||||
*
|
||||
* @param lines The number of lines to scroll. Positive values scroll down, negative values scroll up.
|
||||
* @return True if the scroll was successful, false otherwise
|
||||
*/
|
||||
fun scrollVertically(lines: Int): Boolean
|
||||
|
||||
/**
|
||||
* Scrolls the current line to the top of the display.
|
||||
*
|
||||
* @param line The line number to scroll to (1-based). If 0, uses the current line.
|
||||
* @param start Whether to position the caret at the start of the line
|
||||
* @return True if the scroll was successful, false otherwise
|
||||
*/
|
||||
fun scrollLineToTop(line: Int, start: Boolean): Boolean
|
||||
|
||||
/**
|
||||
* Scrolls the current line to the middle of the display.
|
||||
*
|
||||
* @param line The line number to scroll to (1-based). If 0, uses the current line.
|
||||
* @param start Whether to position the caret at the start of the line
|
||||
* @return True if the scroll was successful, false otherwise
|
||||
*/
|
||||
fun scrollLineToMiddle(line: Int, start: Boolean): Boolean
|
||||
|
||||
/**
|
||||
* Scrolls the current line to the bottom of the display.
|
||||
*
|
||||
* @param line The line number to scroll to (1-based). If 0, uses the current line.
|
||||
* @param start Whether to position the caret at the start of the line
|
||||
* @return True if the scroll was successful, false otherwise
|
||||
*/
|
||||
fun scrollLineToBottom(line: Int, start: Boolean): Boolean
|
||||
|
||||
/**
|
||||
* Scrolls the editor horizontally by a specified number of columns.
|
||||
*
|
||||
* @param columns The number of columns to scroll. Positive values scroll right, negative values scroll left.
|
||||
* @return True if the scroll was successful, false otherwise
|
||||
*/
|
||||
fun scrollHorizontally(columns: Int): Boolean
|
||||
|
||||
/**
|
||||
* Scrolls the editor to position the caret column at the left edge of the display.
|
||||
*
|
||||
* @return True if the scroll was successful, false otherwise
|
||||
*/
|
||||
fun scrollCaretToLeftEdge(): Boolean
|
||||
|
||||
/**
|
||||
* Scrolls the editor to position the caret column at the right edge of the display.
|
||||
*
|
||||
* @return True if the scroll was successful, false otherwise
|
||||
*/
|
||||
fun scrollCaretToRightEdge(): Boolean
|
||||
|
||||
/**
|
||||
* Find the next paragraph-bound offset in the editor.
|
||||
*
|
||||
* @param startLine Line to start the search from.
|
||||
* @param count Search for the [count]-th occurrence.
|
||||
* @param includeWhitespaceLines Should be `true` if we consider lines with whitespaces as empty.
|
||||
* @return next paragraph off
|
||||
*/
|
||||
fun getNextParagraphBoundOffset(startLine: Int, count: Int = 1, includeWhitespaceLines: Boolean = true): Int?
|
||||
|
||||
/**
|
||||
* Finds the next sentence start in the editor from the given offset, based on the specified parameters.
|
||||
*
|
||||
* @param count Search for the [count]-th occurrence.
|
||||
* @param includeCurrent If `true`, includes the current sentence if at its boundary.
|
||||
* @param requireAll If `true`, returns `null` if fewer than [count] sentences are found.
|
||||
* @return The offset of the next sentence start, or `null` if not found or constraints cannot be met.
|
||||
*/
|
||||
fun getNextSentenceStart(
|
||||
startOffset: Int,
|
||||
count: Int = 1,
|
||||
includeCurrent: Boolean,
|
||||
requireAll: Boolean = true,
|
||||
): Int?
|
||||
|
||||
/**
|
||||
* Find the next section in the editor.
|
||||
*
|
||||
* @param startLine The line to start searching from.
|
||||
* @param marker The type of section to find.
|
||||
* @param count Search for the [count]-th occurrence.
|
||||
* @return The offset of the next section.
|
||||
*/
|
||||
fun getNextSectionStart(startLine: Int, marker: Char, count: Int = 1): Int
|
||||
|
||||
/**
|
||||
* Find the start of the previous section in the editor.
|
||||
*
|
||||
* @param startLine The line to start searching from.
|
||||
* @param marker The type of section to find.
|
||||
* @param count Search for the [count]-th occurrence.
|
||||
* @return The offset of the next section.
|
||||
*/
|
||||
fun getPreviousSectionStart(startLine: Int, marker: Char, count: Int = 1): Int
|
||||
|
||||
/**
|
||||
* Find the next sentence end from the given offset.
|
||||
*
|
||||
* @param startOffset The offset to start searching from
|
||||
* @param count Search for the [count]-th occurrence.
|
||||
* @param includeCurrent Whether to count the current position as a sentence end
|
||||
* @param requireAll Whether to require all sentence ends to be found
|
||||
* @return The offset of the next sentence end, or null if not found
|
||||
*/
|
||||
fun getNextSentenceEnd(
|
||||
startOffset: Int,
|
||||
count: Int = 1,
|
||||
includeCurrent: Boolean,
|
||||
requireAll: Boolean = true,
|
||||
): Int?
|
||||
|
||||
/**
|
||||
* Find the next word in the editor's document, from the given starting point
|
||||
*
|
||||
* @param startOffset The offset in the document to search from
|
||||
* @param count Search for the [count]-th occurrence. If negative, search backwards.
|
||||
* @param isBigWord Use WORD instead of word boundaries.
|
||||
* @return The offset of the [count]-th next word, or `null` if not found.
|
||||
*/
|
||||
fun getNextWordStartOffset(startOffset: Int, count: Int = 1, isBigWord: Boolean = false): Int?
|
||||
|
||||
/**
|
||||
* Find the end offset of the next word in the editor's document, from the given starting point
|
||||
*
|
||||
* @param startOffset The offset in the document to search from
|
||||
* @param count Return an offset to the [count] word from the starting position. Will search backwards if negative
|
||||
* @param isBigWord Use WORD instead of word boundaries
|
||||
* @return The offset of the [count] next word/WORD. Will return document bounds if not found
|
||||
*/
|
||||
fun getNextWordEndOffset(startOffset: Int, count: Int = 1, isBigWord: Boolean = false): Int
|
||||
|
||||
/**
|
||||
* Find the next character on the current line
|
||||
*
|
||||
* @param startOffset The offset to start searching from
|
||||
* @param count The number of occurrences to find
|
||||
* @param char The character to find
|
||||
* @return The offset of the next character, or -1 if not found
|
||||
*/
|
||||
fun getNextCharOnLineOffset(startOffset: Int, count: Int = 1, char: Char): Int
|
||||
|
||||
/**
|
||||
* Find the word or WORD at or following the given offset
|
||||
*
|
||||
* Note that if there is no current or following word, the next WORD will be returned. If a WORD is requested, this is
|
||||
* obviously a no-op.
|
||||
*
|
||||
* @param startOffset The offset to search from
|
||||
* @return The range of the word, or null if not found
|
||||
*/
|
||||
fun getWordAtOrFollowingOffset(startOffset: Int, isBigWord: Boolean): Range?
|
||||
|
||||
/**
|
||||
* Returns range of a paragraph containing the given line.
|
||||
*
|
||||
* @param line line to start the search from
|
||||
* @param count search for the count paragraphs forward
|
||||
* @param isOuter true if it is an outer motion, false otherwise
|
||||
* @return the paragraph text range
|
||||
*/
|
||||
fun getParagraphRange(line: Int, count: Int = 1, isOuter: Boolean): Range?
|
||||
|
||||
/**
|
||||
* Find a block quote in the current line
|
||||
*
|
||||
* @param startOffset The offset to start searching from
|
||||
* @param quote The quote character to find
|
||||
* @param isOuter Whether to include the quotes in the range
|
||||
* @return The range of the block quote, or null if not found
|
||||
*/
|
||||
fun getBlockQuoteInLineRange(startOffset: Int, quote: Char, isOuter: Boolean): Range?
|
||||
|
||||
/**
|
||||
* Finds all occurrences of the given pattern within a specified line range.
|
||||
*
|
||||
* @param pattern The Vim-style regex pattern to search for.
|
||||
* @param startLine The line number to start searching from (0-based). Must be within the range [0, lineCount-1].
|
||||
* @param endLine The line number to end searching at (0-based), or -1 for the whole document.
|
||||
* If specified, must be within the range [startLine, lineCount-1].
|
||||
* @param ignoreCase If true, performs case-insensitive search; if false, performs case-sensitive search.
|
||||
* @return A list of Ranges representing all matches found. Empty list if no matches are found.
|
||||
*/
|
||||
fun findAll(pattern: String, startLine: Int, endLine: Int, ignoreCase: Boolean = false): List<Range>
|
||||
|
||||
/**
|
||||
* Finds text matching the given Vim-style regular expression pattern.
|
||||
*
|
||||
* @param pattern The Vim-style regex pattern to search for.
|
||||
* @param startOffset The offset to start searching from. Must be within the range [0, document.length].
|
||||
* @param count Find the [count]-th occurrence of the pattern.
|
||||
* @param backwards If true, search backward from the start offset; if false, search forward.
|
||||
* @return A Range representing the matched text, or null if no match is found.
|
||||
*/
|
||||
fun findPattern(pattern: String, startOffset: Int, count: Int = 1, backwards: Boolean = false): Range?
|
||||
}
|
@@ -1,82 +0,0 @@
|
||||
/*
|
||||
* Copyright 2003-2025 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.api.scopes.editor
|
||||
|
||||
import com.intellij.vim.api.scopes.VimApiDsl
|
||||
import kotlin.contracts.ExperimentalContracts
|
||||
import kotlin.contracts.InvocationKind
|
||||
import kotlin.contracts.contract
|
||||
|
||||
/**
|
||||
* Scope that provides access to editor functions.
|
||||
*/
|
||||
@VimApiDsl
|
||||
abstract class EditorScope {
|
||||
/**
|
||||
* Executes a read-only operation on the editor.
|
||||
*
|
||||
* This function provides access to read-only operations through the [EditorAccessor] interface.
|
||||
* It runs the provided block under a read lock to ensure thread safety when accessing editor state.
|
||||
* The operation is executed asynchronously and returns a [kotlinx.coroutines.Deferred] that can be awaited for the result.
|
||||
*
|
||||
* Example usage:
|
||||
* ```
|
||||
* editor {
|
||||
* val text = read {
|
||||
* text // Access the editor's text content
|
||||
* }.await()
|
||||
* }
|
||||
* ```
|
||||
*
|
||||
* @param block A suspending lambda with [EditorAccessor] receiver that contains the read operations to perform
|
||||
* @return A [kotlinx.coroutines.Deferred] that completes with the result of the block execution
|
||||
*/
|
||||
@OptIn(ExperimentalContracts::class)
|
||||
fun <T> read(block: ReadScope.() -> T): T {
|
||||
contract {
|
||||
callsInPlace(block, InvocationKind.EXACTLY_ONCE)
|
||||
}
|
||||
return this.ideRead(block)
|
||||
}
|
||||
|
||||
/**
|
||||
* Executes a write operation that modifies the editor's state.
|
||||
*
|
||||
* This function provides access to write operations through the [Transaction] interface.
|
||||
* It runs the provided block under a write lock to ensure thread safety when modifying editor state.
|
||||
* The operation is executed asynchronously and returns a [kotlinx.coroutines.Job] that can be joined to wait for completion.
|
||||
*
|
||||
* Example usage:
|
||||
* ```
|
||||
* editor {
|
||||
* val job = change {
|
||||
* // Modify editor content
|
||||
* replaceText(startOffset, endOffset, newText)
|
||||
*
|
||||
* // Add highlights
|
||||
* val highlightId = addHighlight(startOffset, endOffset, backgroundColor, foregroundColor)
|
||||
* }
|
||||
* job.join() // Wait for the changes to complete
|
||||
* }
|
||||
* ```
|
||||
*
|
||||
* @param block A suspending lambda with [Transaction] receiver that contains the write operations to perform
|
||||
* @return A [kotlinx.coroutines.Job] that completes when all write operations are finished
|
||||
*/
|
||||
@OptIn(ExperimentalContracts::class)
|
||||
fun change(block: Transaction.() -> Unit) {
|
||||
contract {
|
||||
callsInPlace(block, InvocationKind.EXACTLY_ONCE)
|
||||
}
|
||||
return ideChange(block)
|
||||
}
|
||||
|
||||
protected abstract fun <T> ideRead(block: ReadScope.() -> T): T
|
||||
protected abstract fun ideChange(block: Transaction.() -> Unit)
|
||||
}
|
@@ -1,81 +0,0 @@
|
||||
/*
|
||||
* Copyright 2003-2025 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.api.scopes.editor
|
||||
|
||||
import com.intellij.vim.api.models.CaretId
|
||||
import com.intellij.vim.api.scopes.VimApiDsl
|
||||
import com.intellij.vim.api.scopes.editor.caret.CaretRead
|
||||
|
||||
/**
|
||||
* Interface that provides functions that open CaretRead scope.
|
||||
*/
|
||||
@VimApiDsl
|
||||
interface ReadScope : EditorAccessor {
|
||||
/**
|
||||
* Executes the provided block for each caret in the editor and returns a list of results.
|
||||
*
|
||||
* This function allows you to perform operations on all carets in the editor in a single call.
|
||||
* The block is executed with each caret as the receiver, and the results are collected into a list.
|
||||
*
|
||||
* Example usage:
|
||||
* ```kotlin
|
||||
* editor {
|
||||
* val caretOffsets = forEachCaret {
|
||||
* offset // Get the offset of each caret
|
||||
* }
|
||||
* // caretOffsets is a List<Int> containing the offset of each caret
|
||||
* }
|
||||
* ```
|
||||
*
|
||||
* @param block The block to execute for each caret
|
||||
* @return A list containing the results of executing the block for each caret
|
||||
*/
|
||||
fun <T> forEachCaret(block: CaretRead.() -> T): List<T>
|
||||
|
||||
/**
|
||||
* Executes the provided block with a specific caret as the receiver.
|
||||
*
|
||||
* This function allows you to perform operations on a specific caret identified by its ID.
|
||||
*
|
||||
* Example usage:
|
||||
* ```kotlin
|
||||
* editor {
|
||||
* with(caretId) {
|
||||
* // Perform operations on the specific caret
|
||||
* val caretOffset = offset
|
||||
* val caretLine = line
|
||||
* }
|
||||
* }
|
||||
* ```
|
||||
*
|
||||
* @param caretId The ID of the caret to use
|
||||
* @param block The block to execute with the specified caret as the receiver
|
||||
*/
|
||||
fun <T> with(caretId: CaretId, block: CaretRead.() -> T): T
|
||||
|
||||
/**
|
||||
* Executes the provided block with the primary caret as the receiver.
|
||||
*
|
||||
* This function allows you to perform operations on the primary caret in the editor.
|
||||
*
|
||||
* Example usage:
|
||||
* ```kotlin
|
||||
* editor {
|
||||
* withPrimaryCaret {
|
||||
* // Perform operations on the primary caret
|
||||
* val primaryCaretOffset = offset
|
||||
* val primaryCaretLine = line
|
||||
* }
|
||||
* }
|
||||
* ```
|
||||
*
|
||||
* @param block The block to execute with the primary caret as the receiver
|
||||
*/
|
||||
fun <T> withPrimaryCaret(block: CaretRead.() -> T): T
|
||||
}
|
@@ -1,198 +0,0 @@
|
||||
/*
|
||||
* Copyright 2003-2025 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.api.scopes.editor
|
||||
|
||||
import com.intellij.vim.api.models.CaretId
|
||||
import com.intellij.vim.api.models.Color
|
||||
import com.intellij.vim.api.models.HighlightId
|
||||
import com.intellij.vim.api.models.Jump
|
||||
import com.intellij.vim.api.scopes.VimApiDsl
|
||||
import com.intellij.vim.api.scopes.editor.caret.CaretTransaction
|
||||
|
||||
/**
|
||||
* Scope for editor functions that should be executed under write lock.
|
||||
*/
|
||||
@VimApiDsl
|
||||
interface Transaction : EditorAccessor {
|
||||
/**
|
||||
* Executes the provided block for each caret in the editor and returns a list of results.
|
||||
*
|
||||
* Example usage:
|
||||
* ```kotlin
|
||||
* editor {
|
||||
* change {
|
||||
* forEachCaret {
|
||||
* // Perform operations on each caret
|
||||
* }
|
||||
* }
|
||||
* }
|
||||
* ```
|
||||
*
|
||||
* @param block The block to execute for each caret
|
||||
* @return A list containing the results of executing the block for each caret
|
||||
*/
|
||||
fun <T> forEachCaret(block: CaretTransaction.() -> T): List<T>
|
||||
|
||||
/**
|
||||
* Executes the provided block with a specific caret as the receiver.
|
||||
*
|
||||
* This function allows you to perform write operations on a specific caret identified by its ID.
|
||||
*
|
||||
* Example usage:
|
||||
* ```kotlin
|
||||
* editor {
|
||||
* change {
|
||||
* val caretId = caretIds.first() // Get the ID of the first caret
|
||||
* with(caretId) {
|
||||
* // Perform operations on the specific caret
|
||||
* deleteText(offset, offset + 5)
|
||||
* updateCaret(newOffset)
|
||||
* }
|
||||
* }
|
||||
* }
|
||||
* ```
|
||||
*
|
||||
* @param caretId The ID of the caret to use
|
||||
* @param block The block to execute with the specified caret as the receiver
|
||||
*/
|
||||
fun <T> with(caretId: CaretId, block: CaretTransaction.() -> T): T
|
||||
|
||||
/**
|
||||
* Executes the provided block with the primary caret as the receiver.
|
||||
*
|
||||
* This function allows you to perform write operations on the primary caret in the editor.
|
||||
*
|
||||
* Example usage:
|
||||
* ```kotlin
|
||||
* editor {
|
||||
* change {
|
||||
* withPrimaryCaret {
|
||||
* // Perform operations on the primary caret
|
||||
* deleteText(offset, offset + 5)
|
||||
* updateCaret(newOffset)
|
||||
* }
|
||||
* }
|
||||
* }
|
||||
* ```
|
||||
*
|
||||
* @param block The block to execute with the primary caret as the receiver
|
||||
*/
|
||||
fun <T> withPrimaryCaret(block: CaretTransaction.() -> T): T
|
||||
|
||||
/**
|
||||
* Adds a new caret at the specified offset in the editor.
|
||||
*
|
||||
* @param offset The offset at which to add the caret
|
||||
* @return The ID of the newly created caret if successful, null otherwise
|
||||
* @throws IllegalArgumentException if offset is not in valid range `[0, fileLength - 1]`
|
||||
*/
|
||||
fun addCaret(offset: Int): CaretId?
|
||||
|
||||
/**
|
||||
* Removes a caret from the editor.
|
||||
*
|
||||
* @param caretId The ID of the caret to remove
|
||||
* @throws IllegalArgumentException if caret with [caretId] is not found
|
||||
*/
|
||||
fun removeCaret(caretId: CaretId)
|
||||
|
||||
/**
|
||||
* Adds a highlight to the editor.
|
||||
*
|
||||
* @param startOffset The start offset of the highlight
|
||||
* @param endOffset The end offset of the highlight
|
||||
* @param backgroundColor The background color of the highlight, or null for no background color
|
||||
* @param foregroundColor The foreground color of the highlight, or null for no foreground color
|
||||
* @return The ID of the newly created highlight
|
||||
*/
|
||||
fun addHighlight(
|
||||
startOffset: Int,
|
||||
endOffset: Int,
|
||||
backgroundColor: Color?,
|
||||
foregroundColor: Color?,
|
||||
): HighlightId
|
||||
|
||||
/**
|
||||
* Removes a highlight from the editor.
|
||||
*
|
||||
* @param highlightId The ID of the highlight to remove
|
||||
*/
|
||||
fun removeHighlight(highlightId: HighlightId)
|
||||
|
||||
/**
|
||||
* Sets a mark at the current position for each caret in the editor.
|
||||
*
|
||||
* @param char The character key of the mark (a-z, A-Z, etc.)
|
||||
* @return True if the mark was successfully set, false otherwise
|
||||
*/
|
||||
fun setMark(char: Char): Boolean
|
||||
|
||||
/**
|
||||
* Removes a mark for all carets in the editor.
|
||||
*
|
||||
* @param char The character key of the mark to remove (a-z, A-Z, etc.)
|
||||
*/
|
||||
fun removeMark(char: Char)
|
||||
|
||||
/**
|
||||
* Sets a global mark at the current position.
|
||||
*
|
||||
* @param char The character key of the mark (A-Z)
|
||||
* @return True if the mark was successfully set, false otherwise
|
||||
*/
|
||||
fun setGlobalMark(char: Char): Boolean
|
||||
|
||||
/**
|
||||
* Removes a global mark.
|
||||
*
|
||||
* @param char The character key of the mark to remove (A-Z)
|
||||
*/
|
||||
fun removeGlobalMark(char: Char)
|
||||
|
||||
/**
|
||||
* Sets a global mark at the specified offset.
|
||||
*
|
||||
* @param char The character key of the mark (A-Z)
|
||||
* @param offset The offset to set the mark to
|
||||
* @return True if the mark was successfully set, false otherwise
|
||||
*/
|
||||
fun setGlobalMark(char: Char, offset: Int): Boolean
|
||||
|
||||
/**
|
||||
* Resets all marks.
|
||||
*
|
||||
* This removes all marks, both global and local.
|
||||
*/
|
||||
fun resetAllMarks()
|
||||
|
||||
/**
|
||||
* Adds a specific jump to the jump list.
|
||||
*
|
||||
* @param jump The jump to add
|
||||
* @param reset Whether to reset the current position in the jump list
|
||||
*/
|
||||
fun addJump(jump: Jump, reset: Boolean = false)
|
||||
|
||||
/**
|
||||
* Removes a jump from the jump list.
|
||||
*
|
||||
* @param jump The jump to remove
|
||||
*/
|
||||
fun removeJump(jump: Jump)
|
||||
|
||||
/**
|
||||
* Removes the last jump from the jump list.
|
||||
*/
|
||||
fun dropLastJump()
|
||||
|
||||
/**
|
||||
* Clears all jumps from the jump list.
|
||||
*/
|
||||
fun clearJumps()
|
||||
}
|
@@ -1,371 +0,0 @@
|
||||
/*
|
||||
* Copyright 2003-2025 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.api.scopes.editor.caret
|
||||
|
||||
import com.intellij.vim.api.models.CaretId
|
||||
import com.intellij.vim.api.models.Line
|
||||
import com.intellij.vim.api.models.Mark
|
||||
import com.intellij.vim.api.models.Range
|
||||
import com.intellij.vim.api.models.TextType
|
||||
import com.intellij.vim.api.scopes.VimApiDsl
|
||||
|
||||
/**
|
||||
* Scope for caret operations that should be executed under the read lock.
|
||||
*/
|
||||
@VimApiDsl
|
||||
interface CaretRead {
|
||||
/**
|
||||
* The unique identifier for this caret.
|
||||
*/
|
||||
val caretId: CaretId
|
||||
|
||||
/**
|
||||
* The current offset (position) of the caret in the document.
|
||||
*/
|
||||
val offset: Int
|
||||
|
||||
/**
|
||||
* The current selection range of the caret.
|
||||
*/
|
||||
val selection: Range
|
||||
|
||||
/**
|
||||
* Information about the current line where the caret is positioned.
|
||||
*/
|
||||
val line: Line
|
||||
|
||||
/**
|
||||
* The last register that was selected for operations.
|
||||
*
|
||||
* Example: After using `"ay` to yank into register 'a', this would return 'a'.
|
||||
* In VimScript, variable `v:register` contains this value.
|
||||
*/
|
||||
val lastSelectedReg: Char
|
||||
|
||||
/**
|
||||
* The default register used when no register is explicitly specified.
|
||||
*
|
||||
* In Vim, this is typically the unnamed register (").
|
||||
*/
|
||||
val defaultRegister: Char
|
||||
|
||||
/**
|
||||
* Indicates whether the current register was explicitly specified by the user.
|
||||
*
|
||||
* Example: After `"ay`, this would be true; after just `y`, this would be false.
|
||||
*/
|
||||
val isRegisterSpecifiedExplicitly: Boolean
|
||||
|
||||
/**
|
||||
* Selects a register for subsequent operations.
|
||||
*
|
||||
* Example: In Vim, pressing `"a` before an operation selects register 'a'.
|
||||
*
|
||||
* @param register The register to select
|
||||
* @return True if the register was successfully selected, false otherwise
|
||||
*/
|
||||
fun selectRegister(register: Char): Boolean
|
||||
|
||||
/**
|
||||
* Resets all registers to their default state.
|
||||
*/
|
||||
fun resetRegisters()
|
||||
|
||||
/**
|
||||
* Checks if a register is writable.
|
||||
*
|
||||
* Some registers in Vim are read-only. Examples of read-only registers:
|
||||
* - ':' (last executed command)
|
||||
* - '.' (last inserted text)
|
||||
* - '/' (last search pattern)
|
||||
*
|
||||
* @param register The register to check
|
||||
* @return True if the register is writable, false otherwise
|
||||
*/
|
||||
fun isWritable(register: Char): Boolean
|
||||
|
||||
/**
|
||||
* Checks if a register is connected to the system clipboard.
|
||||
*
|
||||
* In Vim, registers '+' and '*' are connected to the system clipboard.
|
||||
* Example: Using `"+y` yanks text to the system clipboard.
|
||||
*
|
||||
* @param register The register to check
|
||||
* @return True if the register is connected to the system clipboard, false otherwise
|
||||
*/
|
||||
fun isSystemClipboard(register: Char): Boolean
|
||||
|
||||
/**
|
||||
* Checks if the primary selection register is supported.
|
||||
*
|
||||
* Example: On Linux, using `"*y` yanks text to the primary selection.
|
||||
*
|
||||
* @return True if the primary selection register is supported, false otherwise
|
||||
*/
|
||||
fun isPrimaryRegisterSupported(): Boolean
|
||||
|
||||
/**
|
||||
* The marks for the current visual selection.
|
||||
*
|
||||
* In Vim, these are the '< and '> marks.
|
||||
* Example: After making a visual selection and then pressing ESC, `'<` and `'>` mark the beginning and end.
|
||||
* In VimScript `getpos("'<")` and `getpos("'>")` are used to get these marks.
|
||||
*/
|
||||
val selectionMarks: Range?
|
||||
|
||||
/**
|
||||
* The marks for the last change.
|
||||
*
|
||||
* In Vim, these are the '[ and '] marks.
|
||||
* Example: After a change operation like `cw`, these marks indicate the changed region.
|
||||
* In VimScript, `getpos("'[")` and `getpos("']")` are used to get these marks.
|
||||
*/
|
||||
val changeMarks: Range?
|
||||
|
||||
/**
|
||||
* Gets the text content of a register.
|
||||
*
|
||||
* Example: In Vim, `:echo @a` shows the content of register 'a'.
|
||||
* In VimScript `getreg('a')` is used to get the content of register 'a'.
|
||||
*
|
||||
* @param register The register to get text from
|
||||
* @return The text content of the register, or null if the register is empty or doesn't exist
|
||||
*/
|
||||
fun getReg(register: Char): String?
|
||||
|
||||
/**
|
||||
* Gets the type of text stored in a register (character-wise, line-wise, or block-wise).
|
||||
*
|
||||
* In VimScript, `getregtype('a')` is used to get the type of register 'a'.
|
||||
*
|
||||
* @param register The register to get the type from
|
||||
* @return The type of text in the register, or null if the register is empty or doesn't exist
|
||||
*/
|
||||
fun getRegType(register: Char): TextType?
|
||||
|
||||
/**
|
||||
* Sets the text content and type of a register.
|
||||
*
|
||||
* In VimScript, `setreg('a', 'text', 'c')` is used to set register 'a' to "text" with character-wise type.
|
||||
*
|
||||
* @param register The register to set
|
||||
* @param text The text to store in the register
|
||||
* @param textType The type of text (character-wise, line-wise, or block-wise)
|
||||
* @return True if the register was successfully set, false otherwise
|
||||
*/
|
||||
fun setReg(register: Char, text: String, textType: TextType = TextType.CHARACTER_WISE): Boolean
|
||||
|
||||
/**
|
||||
* Gets a mark by its character key for the current caret.
|
||||
*
|
||||
* @param char The character key of the mark (a-z, 0-9, etc.)
|
||||
* @return The mark, or null if the mark doesn't exist
|
||||
*/
|
||||
fun getMark(char: Char): Mark?
|
||||
|
||||
/**
|
||||
* All local marks for the current caret.
|
||||
*/
|
||||
val localMarks: Set<Mark>
|
||||
|
||||
/**
|
||||
* Sets a mark at the current caret position.
|
||||
*
|
||||
* @param char The character key of the mark (a-z, etc.)
|
||||
* @return True if the mark was successfully set, false otherwise
|
||||
*/
|
||||
fun setMark(char: Char): Boolean
|
||||
|
||||
/**
|
||||
* Sets a mark at the specified offset.
|
||||
*
|
||||
* @param char The character key of the mark (a-z, etc.)
|
||||
* @param offset The offset to set the mark to
|
||||
* @return True if the mark was successfully set, false otherwise
|
||||
*/
|
||||
fun setMark(char: Char, offset: Int): Boolean
|
||||
|
||||
/**
|
||||
* Removes a local mark for the current caret.
|
||||
*
|
||||
* @param char The character key of the mark to remove (a-z, etc.)
|
||||
*/
|
||||
fun removeLocalMark(char: Char)
|
||||
|
||||
/**
|
||||
* Resets all marks for the current caret.
|
||||
*/
|
||||
fun resetAllMarksForCaret()
|
||||
|
||||
/**
|
||||
* Scrolls a full page up or down.
|
||||
*
|
||||
* @param pages The number of pages to scroll. Positive values scroll down, negative values scroll up.
|
||||
* @return True if the scroll was successful, false otherwise
|
||||
*/
|
||||
fun scrollFullPage(pages: Int): Boolean
|
||||
|
||||
/**
|
||||
* Scrolls half a page up.
|
||||
*
|
||||
* @param lines The number of lines to scroll.
|
||||
* @return True if the scroll was successful, false otherwise
|
||||
*/
|
||||
fun scrollHalfPageUp(lines: Int): Boolean
|
||||
|
||||
/**
|
||||
* Scrolls half a page up.
|
||||
*
|
||||
* @param lines The number of lines to scroll.
|
||||
* @return True if the scroll was successful, false otherwise
|
||||
*/
|
||||
fun scrollHalfPageDown(lines: Int): Boolean
|
||||
|
||||
/**
|
||||
* Selects a window in the same row as the current window.
|
||||
*
|
||||
* @param relativePosition The relative position of the window to select.
|
||||
* Positive values select windows to the right,
|
||||
* negative values select windows to the left.
|
||||
*/
|
||||
fun selectWindowHorizontally(relativePosition: Int)
|
||||
|
||||
/**
|
||||
* Selects a window in the same column as the current window.
|
||||
*
|
||||
* @param relativePosition The relative position of the window to select.
|
||||
* Positive values select the windows below,
|
||||
* negative values select the windows above.
|
||||
*/
|
||||
fun selectWindowInVertically(relativePosition: Int)
|
||||
|
||||
/**
|
||||
* Finds the offset of the next paragraph boundary.
|
||||
*
|
||||
* @param count Search for the [count]-th occurrence.
|
||||
* @param includeWhitespaceLines Should be `true` if we consider lines with whitespaces as empty.
|
||||
* @return next paragraph off
|
||||
*/
|
||||
fun getNextParagraphBoundOffset(count: Int = 1, includeWhitespaceLines: Boolean = true): Int?
|
||||
|
||||
/**
|
||||
* Finds the next sentence start in the editor from the given offset, based on the specified parameters.
|
||||
*
|
||||
* @param count Search for the [count]-th occurrence.
|
||||
* @param includeCurrent If `true`, includes the current sentence if at its boundary.
|
||||
* @param requireAll If `true`, returns `null` if fewer than [count] sentences are found.
|
||||
* @return The offset of the next sentence start, or `null` if not found or constraints cannot be met.
|
||||
*/
|
||||
fun getNextSentenceStart(count: Int = 1, includeCurrent: Boolean, requireAll: Boolean = true): Int?
|
||||
|
||||
/**
|
||||
* Find the next section in the editor.
|
||||
*
|
||||
* @param marker The type of section to find.
|
||||
* @param count Search for the [count]-th occurrence.
|
||||
* @return The offset of the next section.
|
||||
*/
|
||||
fun getNextSectionStart(marker: Char, count: Int = 1): Int
|
||||
|
||||
/**
|
||||
* Find the start of the previous section in the editor.
|
||||
*
|
||||
* @param marker The type of section to find.
|
||||
* @param count Search for the [count]-th occurrence.
|
||||
* @return The offset of the next section.
|
||||
*/
|
||||
fun getPreviousSectionStart(marker: Char, count: Int = 1): Int
|
||||
|
||||
/**
|
||||
* Finds the end offset of the next sentence from the current caret position.
|
||||
*
|
||||
* @param count Search for the [count]-th occurrence.
|
||||
* @param includeCurrent Whether to count the current position as a sentence end
|
||||
* @param requireAll Whether to require all sentence ends to be found
|
||||
* @return The offset of the next sentence end, or null if not found
|
||||
*/
|
||||
fun getNextSentenceEnd(count: Int = 1, includeCurrent: Boolean, requireAll: Boolean = true): Int?
|
||||
|
||||
/**
|
||||
* Finds the end offset of the next method from the current caret position.
|
||||
*
|
||||
* @param count Search for the [count]-th occurrence.
|
||||
* @return The offset of the end of the next method.
|
||||
*/
|
||||
fun getMethodEndOffset(count: Int = 1): Int
|
||||
|
||||
/**
|
||||
* Finds the start offset of the next method from the current caret position.
|
||||
*
|
||||
* @param count Search for the [count]-th occurrence.
|
||||
* @return The offset of the start of the next method.
|
||||
*/
|
||||
fun getMethodStartOffset(count: Int = 1): Int
|
||||
|
||||
/**
|
||||
* Finds the next occurrence of a specific character on the current line.
|
||||
*
|
||||
* @param count Search for the [count]-th occurrence.
|
||||
* @param char The character to find.
|
||||
* @return The offset of the found character, or -1 if not found.
|
||||
*/
|
||||
fun getNextCharOnLineOffset(count: Int = 1, char: Char): Int
|
||||
|
||||
/**
|
||||
* Finds the word at or following the current caret position.
|
||||
*
|
||||
* Note that if no word/WORD is found at or following the caret on the current line, the WORD at or following is
|
||||
* always returned.
|
||||
*
|
||||
* @param isBigWord Search for word or WORD.
|
||||
* @return A Range representing the found word, or null if no word is found.
|
||||
*/
|
||||
fun getCurrentOrFollowingWord(isBigWord: Boolean): Range?
|
||||
|
||||
/**
|
||||
* Find the range of the word text object at the location of the caret
|
||||
*/
|
||||
fun getWordTextObjectRange(count: Int = 1, isOuter: Boolean, isBigWord: Boolean): Range
|
||||
|
||||
/**
|
||||
* Find the range of the sentence text object at the location of the caret
|
||||
*/
|
||||
fun getSentenceRange(count: Int = 1, isOuter: Boolean): Range
|
||||
|
||||
/**
|
||||
* Returns range of a paragraph containing the caret
|
||||
*
|
||||
* @param count Search for the [count]-th occurrence.
|
||||
* @param isOuter true if it is an outer motion, false otherwise
|
||||
* @return the paragraph text range
|
||||
*/
|
||||
fun getParagraphRange(count: Int = 1, isOuter: Boolean): Range?
|
||||
|
||||
/**
|
||||
* Find the range of a block tag at the location of the caret
|
||||
*/
|
||||
fun getBlockTagRange(count: Int = 1, isOuter: Boolean): Range?
|
||||
|
||||
/**
|
||||
* Find a block quote in the current line at the location of the caret
|
||||
*
|
||||
* @param quote The quote character to find
|
||||
* @param isOuter Whether to include the quotes in the range
|
||||
* @return The range of the block quote, or null if not found
|
||||
*/
|
||||
fun getBlockQuoteInLineRange(quote: Char, isOuter: Boolean): Range?
|
||||
|
||||
/**
|
||||
* Finds the offset of the next misspelled word from the current caret position.
|
||||
*
|
||||
* @param count Search for the [count]-th occurrence.
|
||||
* @return The offset of the next misspelled word.
|
||||
*/
|
||||
fun getNextMisspelledWordOffset(count: Int = 1): Int
|
||||
}
|
@@ -1,121 +0,0 @@
|
||||
/*
|
||||
* Copyright 2003-2025 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.api.scopes.editor.caret
|
||||
|
||||
import com.intellij.vim.api.models.Range
|
||||
import com.intellij.vim.api.scopes.VimApiDsl
|
||||
import com.intellij.vim.api.scopes.editor.EditorAccessor
|
||||
|
||||
/**
|
||||
* Scope for caret operations that should be executed under the write lock.
|
||||
*/
|
||||
@VimApiDsl
|
||||
interface CaretTransaction : CaretRead, EditorAccessor {
|
||||
/**
|
||||
* Updates the caret position and optionally sets a selection.
|
||||
*
|
||||
* If a selection is provided, the caret will have this selection after moving to the new offset.
|
||||
* If no selection is provided, any existing selection will be removed.
|
||||
*
|
||||
* The selection range is exclusive, meaning that the character at the end offset is not
|
||||
* included in the selection. For example, a selection of (0, 3) would select the first
|
||||
* three characters of the text.
|
||||
*
|
||||
* @param offset The new offset (position) for the caret
|
||||
* @param selection Optional selection range
|
||||
* @throws IllegalArgumentException If the offset is not in the valid range [0, fileSize),
|
||||
* or if the selection range is invalid (start or end out of range,
|
||||
* or start > end)
|
||||
*/
|
||||
fun updateCaret(offset: Int, selection: Range.Simple? = null)
|
||||
|
||||
/**
|
||||
* Inserts text at the specified position in the document.
|
||||
*
|
||||
* @param position The position (offset) where the text should be inserted
|
||||
* (a zero-base character offset from the start of the document)
|
||||
* @param text The text to insert
|
||||
* @param caretAtEnd If true (default), places the caret after on the last character of the inserted text;
|
||||
* if false, places the caret at the beginning of the inserted text
|
||||
* @param insertBeforeCaret If true, inserts the text before the specified position;
|
||||
* if false (default), inserts the text at the specified position
|
||||
* @return true if the insertion was successful, false otherwise
|
||||
* @throws IllegalArgumentException If the position is not in the valid range [0, fileSize)
|
||||
*/
|
||||
fun insertText(
|
||||
position: Int,
|
||||
text: String,
|
||||
caretAtEnd: Boolean = true,
|
||||
insertBeforeCaret: Boolean = false,
|
||||
): Boolean
|
||||
|
||||
/**
|
||||
* Replaces the text between startOffset (inclusive) and endOffset (exclusive)
|
||||
* with the specified text. After the operation, the caret is positioned before the last
|
||||
* character of the replaced text.
|
||||
*
|
||||
* @param startOffset The start offset (inclusive) of the text to be replaced
|
||||
* @param endOffset The end offset (exclusive) of the text to be replaced
|
||||
* @param text The new text to replace the existing text
|
||||
* @return true if the replacement was successful, false otherwise
|
||||
* @throws IllegalArgumentException If the offsets are not in the valid range [0, fileSize),
|
||||
* or if startOffset > endOffset
|
||||
*/
|
||||
fun replaceText(
|
||||
startOffset: Int,
|
||||
endOffset: Int,
|
||||
text: String,
|
||||
): Boolean
|
||||
|
||||
/**
|
||||
* Replaces text in multiple ranges (blocks) with new text.
|
||||
*
|
||||
* This function performs a blockwise replacement, replacing each range in the block
|
||||
* with the corresponding string from the text list. The number of replacement strings
|
||||
* must match the number of ranges in the block.
|
||||
*
|
||||
* @param range A block of ranges to be replaced
|
||||
* @param text A list of strings to replace each range in the block
|
||||
* @throws IllegalArgumentException If the size of the text list doesn't match the number of ranges in the block,
|
||||
* or if any range in the block is invalid
|
||||
*/
|
||||
fun replaceTextBlockwise(
|
||||
range: Range.Block,
|
||||
text: List<String>,
|
||||
)
|
||||
|
||||
/**
|
||||
* Deletes text between the specified offsets.
|
||||
*
|
||||
* This function deletes the text between startOffset (inclusive) and endOffset (exclusive).
|
||||
* If startOffset equals endOffset, no text is deleted.
|
||||
* If startOffset > endOffset, the implementation swaps them and deletes the text between them.
|
||||
*
|
||||
* @param startOffset The start offset (inclusive) of the text to be deleted
|
||||
* @param endOffset The end offset (exclusive) of the text to be deleted
|
||||
* @return true if the deletion was successful, false otherwise
|
||||
* @throws Exception If endOffset is beyond the file size
|
||||
*/
|
||||
fun deleteText(
|
||||
startOffset: Int,
|
||||
endOffset: Int,
|
||||
): Boolean
|
||||
|
||||
/**
|
||||
* Adds a jump with the current caret's position to the jump list.
|
||||
*
|
||||
* @param reset Whether to reset the current position in the jump list
|
||||
*/
|
||||
fun addJump(reset: Boolean)
|
||||
|
||||
/**
|
||||
* Saves the location of the current caret to the jump list and sets the ' mark.
|
||||
*/
|
||||
fun saveJumpLocation()
|
||||
}
|
564
build.gradle.kts
564
build.gradle.kts
@@ -31,13 +31,11 @@ import kotlinx.serialization.json.putJsonObject
|
||||
import org.eclipse.jgit.api.Git
|
||||
import org.eclipse.jgit.lib.RepositoryBuilder
|
||||
import org.intellij.markdown.ast.getTextInNode
|
||||
import org.intellij.markdown.ast.impl.ListCompositeNode
|
||||
import org.jetbrains.changelog.Changelog
|
||||
import org.jetbrains.intellij.platform.gradle.TestFrameworkType
|
||||
import org.jetbrains.intellij.platform.gradle.tasks.aware.SplitModeAware
|
||||
import org.jetbrains.kotlin.gradle.dsl.JvmTarget
|
||||
import org.jetbrains.kotlin.gradle.dsl.KotlinVersion
|
||||
import org.jetbrains.changelog.exceptions.MissingVersionException
|
||||
import org.kohsuke.github.GHUser
|
||||
import java.net.HttpURLConnection
|
||||
import java.net.URL
|
||||
|
||||
buildscript {
|
||||
repositories {
|
||||
@@ -46,19 +44,19 @@ buildscript {
|
||||
}
|
||||
|
||||
dependencies {
|
||||
classpath("org.jetbrains.kotlin:kotlin-gradle-plugin:2.2.0")
|
||||
classpath("org.jetbrains.kotlin:kotlin-gradle-plugin:1.8.21")
|
||||
classpath("com.github.AlexPl292:mark-down-to-slack:1.1.2")
|
||||
classpath("org.eclipse.jgit:org.eclipse.jgit:6.6.0.202305301015-r")
|
||||
|
||||
// This is needed for jgit to connect to ssh
|
||||
classpath("org.eclipse.jgit:org.eclipse.jgit.ssh.apache:7.3.0.202506031305-r")
|
||||
classpath("org.eclipse.jgit:org.eclipse.jgit.ssh.apache:6.7.0.202309050840-r")
|
||||
classpath("org.kohsuke:github-api:1.305")
|
||||
|
||||
classpath("io.ktor:ktor-client-core:3.3.0")
|
||||
classpath("io.ktor:ktor-client-cio:3.3.0")
|
||||
classpath("io.ktor:ktor-client-auth:3.3.0")
|
||||
classpath("io.ktor:ktor-client-content-negotiation:3.3.0")
|
||||
classpath("io.ktor:ktor-serialization-kotlinx-json:3.3.0")
|
||||
classpath("io.ktor:ktor-client-core:2.3.4")
|
||||
classpath("io.ktor:ktor-client-cio:2.3.4")
|
||||
classpath("io.ktor:ktor-client-auth:2.3.4")
|
||||
classpath("io.ktor:ktor-client-content-negotiation:2.3.4")
|
||||
classpath("io.ktor:ktor-serialization-kotlinx-json:2.3.4")
|
||||
|
||||
// This comes from the changelog plugin
|
||||
// classpath("org.jetbrains:markdown:0.3.1")
|
||||
@@ -66,31 +64,43 @@ buildscript {
|
||||
}
|
||||
|
||||
plugins {
|
||||
antlr
|
||||
java
|
||||
kotlin("jvm") version "2.2.0"
|
||||
kotlin("jvm") version "1.8.21"
|
||||
application
|
||||
id("java-test-fixtures")
|
||||
|
||||
// NOTE: Unignore "test block comment falls back to line comment when not available" test
|
||||
// After changing this version. It supposed to work on the next version of the gradle plugin
|
||||
// Or go report to the devs that this test still fails.
|
||||
id("org.jetbrains.intellij.platform") version "2.9.0"
|
||||
id("org.jetbrains.intellij") version "1.15.0"
|
||||
id("org.jetbrains.changelog") version "2.2.0"
|
||||
|
||||
// ktlint linter - read more: https://github.com/JLLeitschuh/ktlint-gradle
|
||||
// id("org.jlleitschuh.gradle.ktlint") version "11.3.1"
|
||||
|
||||
id("org.jetbrains.changelog") version "2.4.0"
|
||||
id("org.jetbrains.kotlinx.kover") version "0.6.1"
|
||||
id("com.dorongold.task-tree") version "4.0.1"
|
||||
id("com.google.devtools.ksp") version "2.2.0-2.0.2"
|
||||
id("com.dorongold.task-tree") version "2.1.1"
|
||||
|
||||
id("com.google.devtools.ksp") version "1.8.21-1.0.11"
|
||||
}
|
||||
|
||||
val moduleSources by configurations.registering
|
||||
ksp {
|
||||
arg("generated_directory", "$projectDir/src/main/resources")
|
||||
arg("vimscript_functions_file", "intellij_vimscript_functions.json")
|
||||
arg("ex_commands_file", "intellij_ex_commands.json")
|
||||
}
|
||||
|
||||
afterEvaluate {
|
||||
// tasks.named("kspKotlin").configure { dependsOn("clean") }
|
||||
tasks.named("kspKotlin").configure { dependsOn("generateGrammarSource") }
|
||||
tasks.named("kspTestKotlin").configure { enabled = false }
|
||||
}
|
||||
|
||||
// Import variables from gradle.properties file
|
||||
val javaVersion: String by project
|
||||
val kotlinVersion: String by project
|
||||
val ideaVersion: String by project
|
||||
val ideaType: String by project
|
||||
val downloadIdeaSources: String by project
|
||||
val instrumentPluginCode: String by project
|
||||
val remoteRobotVersion: String by project
|
||||
val antlrVersion: String by project
|
||||
|
||||
val publishChannels: String by project
|
||||
val publishToken: String by project
|
||||
@@ -98,92 +108,41 @@ val publishToken: String by project
|
||||
val slackUrl: String by project
|
||||
val youtrackToken: String by project
|
||||
|
||||
val releaseType: String? by project
|
||||
|
||||
repositories {
|
||||
mavenCentral()
|
||||
intellijPlatform {
|
||||
defaultRepositories()
|
||||
}
|
||||
maven { url = uri("https://cache-redirector.jetbrains.com/intellij-dependencies") }
|
||||
}
|
||||
|
||||
dependencies {
|
||||
api(project(":vim-engine"))
|
||||
api(project(":api"))
|
||||
ksp(project(":annotation-processors"))
|
||||
compileOnly(project(":annotation-processors"))
|
||||
|
||||
compileOnly("org.jetbrains.kotlin:kotlin-stdlib:$kotlinVersion")
|
||||
compileOnly("org.jetbrains:annotations:26.0.2-1")
|
||||
|
||||
intellijPlatform {
|
||||
// Snapshots don't use installers
|
||||
// https://plugins.jetbrains.com/docs/intellij/tools-intellij-platform-gradle-plugin-dependencies-extension.html#target-versions-installers
|
||||
var useInstaller = "EAP-SNAPSHOT" !in ideaVersion
|
||||
if (ideaType == "RD") {
|
||||
// Using Rider as a target IntelliJ Platform with `useInstaller = true` is currently not supported, please set `useInstaller = false` instead. See: https://github.com/JetBrains/intellij-platform-gradle-plugin/issues/1852
|
||||
useInstaller = false
|
||||
}
|
||||
|
||||
// Note that it is also possible to use local("...") to compile against a locally installed IDE
|
||||
// E.g. local("/Users/{user}/Applications/IntelliJ IDEA Ultimate.app")
|
||||
// Or something like: intellijIdeaUltimate(ideaVersion)
|
||||
create(ideaType, ideaVersion) { this.useInstaller = useInstaller }
|
||||
|
||||
pluginVerifier()
|
||||
zipSigner()
|
||||
|
||||
testFramework(TestFrameworkType.Platform)
|
||||
testFramework(TestFrameworkType.JUnit5)
|
||||
|
||||
// AceJump is an optional dependency. We use their SessionManager class to check if it's active
|
||||
plugin("AceJump", "3.8.19")
|
||||
|
||||
bundledPlugins("org.jetbrains.plugins.terminal")
|
||||
|
||||
// VERSION UPDATE: This module is required since 2025.2
|
||||
if (ideaVersion == "LATEST-EAP-SNAPSHOT") {
|
||||
bundledModule("intellij.spellchecker")
|
||||
}
|
||||
if (ideaVersion.startsWith("2025.2")) {
|
||||
bundledModule("intellij.spellchecker")
|
||||
}
|
||||
if (ideaVersion.startsWith("2025.3")) {
|
||||
bundledModule("intellij.spellchecker")
|
||||
}
|
||||
}
|
||||
|
||||
moduleSources(project(":vim-engine", "sourcesJarArtifacts"))
|
||||
|
||||
// --------- Test dependencies ----------
|
||||
|
||||
testApi("com.squareup.okhttp3:okhttp:5.0.0")
|
||||
compileOnly("org.jetbrains:annotations:24.0.1")
|
||||
|
||||
// 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:core-rpc:0.2.3")
|
||||
testFixturesImplementation("com.ensarsarajcic.neovim.java:neovim-api:0.2.3")
|
||||
testFixturesImplementation("com.ensarsarajcic.neovim.java:core-rpc:0.2.3")
|
||||
|
||||
// https://mvnrepository.com/artifact/org.jetbrains.kotlin/kotlin-test
|
||||
testImplementation("org.jetbrains.kotlin:kotlin-test:$kotlinVersion")
|
||||
testFixturesImplementation("org.jetbrains.kotlin:kotlin-test:$kotlinVersion")
|
||||
|
||||
// https://mvnrepository.com/artifact/org.mockito.kotlin/mockito-kotlin
|
||||
testImplementation("org.mockito.kotlin:mockito-kotlin:6.1.0")
|
||||
testImplementation("org.mockito.kotlin:mockito-kotlin:5.1.0")
|
||||
|
||||
testImplementation("org.junit.jupiter:junit-jupiter-api:6.0.0")
|
||||
testImplementation("org.junit.jupiter:junit-jupiter-engine:6.0.0")
|
||||
testImplementation("org.junit.jupiter:junit-jupiter-params:6.0.0")
|
||||
testFixturesImplementation("org.junit.jupiter:junit-jupiter-api:6.0.0")
|
||||
testFixturesImplementation("org.junit.jupiter:junit-jupiter-engine:6.0.0")
|
||||
testFixturesImplementation("org.junit.jupiter:junit-jupiter-params:6.0.0")
|
||||
testImplementation("com.intellij.remoterobot:remote-robot:$remoteRobotVersion")
|
||||
testImplementation("com.intellij.remoterobot:remote-fixtures:$remoteRobotVersion")
|
||||
testImplementation("com.automation-remarks:video-recorder-junit:2.0")
|
||||
runtimeOnly("org.antlr:antlr4-runtime:$antlrVersion")
|
||||
antlr("org.antlr:antlr4:$antlrVersion")
|
||||
|
||||
// Temp workaround suggested in https://plugins.jetbrains.com/docs/intellij/tools-intellij-platform-gradle-plugin-faq.html#junit5-test-framework-refers-to-junit4
|
||||
// Can be removed when IJPL-159134 is fixed
|
||||
// testRuntimeOnly("junit:junit:4.13.2")
|
||||
testImplementation("org.junit.vintage:junit-vintage-engine:6.0.0")
|
||||
// testFixturesImplementation("org.junit.vintage:junit-vintage-engine:5.10.3")
|
||||
api(project(":vim-engine"))
|
||||
|
||||
ksp(project(":annotation-processors"))
|
||||
compileOnly(project(":annotation-processors"))
|
||||
|
||||
testApi("com.squareup.okhttp3:okhttp:4.11.0")
|
||||
|
||||
testImplementation("org.junit.jupiter:junit-jupiter-api:5.10.0")
|
||||
testImplementation("org.junit.jupiter:junit-jupiter-engine:5.10.0")
|
||||
testImplementation("org.junit.jupiter:junit-jupiter-params:5.10.0")
|
||||
}
|
||||
|
||||
configurations {
|
||||
@@ -192,106 +151,90 @@ configurations {
|
||||
}
|
||||
}
|
||||
|
||||
val currentJavaVersion = javaToolchains.launcherFor {}.get().metadata.languageVersion.toString()
|
||||
if (currentJavaVersion != javaVersion) {
|
||||
// NOTE: I made this exception because the default Gradle error message is horrible, noone can understand it.
|
||||
throw RuntimeException(
|
||||
"""
|
||||
Incorrect java version used for building.
|
||||
IdeaVim uses java version $javaVersion, but the current java version is $currentJavaVersion.
|
||||
If IntelliJ IDEA is used, change the setting in "Settings | Build, Execution, Deployment | Build Tools | Gradle"
|
||||
If build is run from the terminal, set JAVA_HOME environment variable to the correct java version.
|
||||
""".trimIndent()
|
||||
)
|
||||
// --- Compilation
|
||||
// This can be moved to other test registration when issue with tests in gradle will be fixed
|
||||
tasks.register<Test>("testWithNeovim") {
|
||||
group = "verification"
|
||||
systemProperty("ideavim.nvim.test", "true")
|
||||
exclude("/ui/**")
|
||||
exclude("**/longrunning/**")
|
||||
exclude("**/propertybased/**")
|
||||
useJUnitPlatform()
|
||||
}
|
||||
|
||||
tasks.register<Test>("testPropertyBased") {
|
||||
group = "verification"
|
||||
// include("**/propertybased/**")
|
||||
useJUnitPlatform()
|
||||
}
|
||||
|
||||
tasks.register<Test>("testLongRunning") {
|
||||
group = "verification"
|
||||
// include("**/longrunning/**")
|
||||
useJUnitPlatform()
|
||||
}
|
||||
|
||||
tasks {
|
||||
test {
|
||||
useJUnitPlatform()
|
||||
// Issue in gradle 7.3
|
||||
val test by getting(Test::class) {
|
||||
isScanForTestClasses = false
|
||||
// Only run tests from classes that end with "Test"
|
||||
include("**/*Test.class")
|
||||
include("**/*test.class")
|
||||
include("**/*Tests.class")
|
||||
exclude("**/ParserTest.class")
|
||||
}
|
||||
|
||||
// Set teamcity env variable locally to run additional tests for leaks.
|
||||
println("Project leak checks: If you experience project leaks on TeamCity that doesn't reproduce locally")
|
||||
println("Uncomment the following line in build.gradle to enable leak checks (see build.gradle config)")
|
||||
// environment("TEAMCITY_VERSION" to "X")
|
||||
val testWithNeovim by getting(Test::class) {
|
||||
isScanForTestClasses = false
|
||||
// Only run tests from classes that end with "Test"
|
||||
include("**/*Test.class")
|
||||
include("**/*test.class")
|
||||
include("**/*Tests.class")
|
||||
exclude("**/ParserTest.class")
|
||||
exclude("**/longrunning/**")
|
||||
exclude("**/propertybased/**")
|
||||
}
|
||||
|
||||
systemProperty("ideavim.nvim.test", System.getProperty("nvim") ?: false)
|
||||
val testPropertyBased by getting(Test::class) {
|
||||
isScanForTestClasses = false
|
||||
// Only run tests from classes that end with "Test"
|
||||
include("**/propertybased/*Test.class")
|
||||
include("**/propertybased/*test.class")
|
||||
include("**/propertybased/*Tests.class")
|
||||
}
|
||||
|
||||
// This removes all localization plugins from the test version of IJ.
|
||||
// There is a bug that IJ for tests may be loaded with a different locale and some keys may be missing there,
|
||||
// what breaks the tests. This usually happens in EAP versions of IJ.
|
||||
classpath -= classpath.filter { it.name.startsWith("localization-") && it.name.endsWith(".jar") }
|
||||
val testLongRunning by getting(Test::class) {
|
||||
isScanForTestClasses = false
|
||||
// Only run tests from classes that end with "Test"
|
||||
include("**/longrunning/**/*Test.class")
|
||||
include("**/longrunning/**/*test.class")
|
||||
include("**/longrunning/**/*Tests.class")
|
||||
exclude("**/longrunning/**/ParserTest.class")
|
||||
}
|
||||
|
||||
compileJava {
|
||||
// CodeQL can't resolve the 'by project' property, so we need to give it a hint. This is the minimum version we need
|
||||
// so doesn't have to match exactly
|
||||
// Hint for the CodeQL autobuilder: sourceCompatibility = 17
|
||||
sourceCompatibility = javaVersion
|
||||
targetCompatibility = javaVersion
|
||||
|
||||
options.encoding = "UTF-8"
|
||||
}
|
||||
|
||||
// Note that this will run the plugin installed in the IDE specified in dependencies. To run in a different IDE, use
|
||||
// a custom task (see below)
|
||||
runIde {
|
||||
systemProperty("octopus.handler", System.getProperty("octopus.handler") ?: true)
|
||||
}
|
||||
|
||||
// Uncomment to run the plugin in a custom IDE, rather than the IDE specified as a compile target in dependencies
|
||||
// Note that the version must be greater than the plugin's target version, for obvious reasons
|
||||
// You can also set splitMode and splitModeTarget here to test split mode in a custom IDE
|
||||
// val runIdeCustom by intellijPlatformTesting.runIde.registering {
|
||||
// type = IntelliJPlatformType.Rider
|
||||
// version = "2024.1.2"
|
||||
// }
|
||||
|
||||
// Uncomment to run the plugin in a locally installed IDE
|
||||
// val runIdeLocal by intellijPlatformTesting.runIde.registering {
|
||||
// localPath = file("/Users/{user}/Applications/WebStorm.app")
|
||||
// }
|
||||
|
||||
val runIdeForUiTests by intellijPlatformTesting.runIde.registering {
|
||||
task {
|
||||
jvmArgumentProviders += CommandLineArgumentProvider {
|
||||
listOf(
|
||||
"-Drobot-server.port=8082",
|
||||
"-Dide.mac.message.dialogs.as.sheets=false",
|
||||
"-Djb.privacy.policy.text=<!--999.999-->",
|
||||
"-Djb.consents.confirmation.enabled=false",
|
||||
"-Dide.show.tips.on.startup.default.value=false",
|
||||
"-Doctopus.handler=" + (System.getProperty("octopus.handler") ?: true),
|
||||
)
|
||||
compileKotlin {
|
||||
kotlinOptions {
|
||||
jvmTarget = javaVersion
|
||||
apiVersion = "1.6"
|
||||
freeCompilerArgs = listOf("-Xjvm-default=all-compatibility")
|
||||
// allWarningsAsErrors = true
|
||||
}
|
||||
}
|
||||
|
||||
plugins {
|
||||
robotServerPlugin(remoteRobotVersion)
|
||||
compileTestKotlin {
|
||||
kotlinOptions {
|
||||
jvmTarget = javaVersion
|
||||
apiVersion = "1.6"
|
||||
// allWarningsAsErrors = true
|
||||
}
|
||||
}
|
||||
|
||||
val runIdeSplitMode by intellijPlatformTesting.runIde.registering {
|
||||
splitMode = true
|
||||
splitModeTarget = SplitModeAware.SplitModeTarget.FRONTEND
|
||||
}
|
||||
|
||||
// Add plugin open API sources to the plugin ZIP
|
||||
val sourcesJar by registering(Jar::class) {
|
||||
dependsOn(moduleSources)
|
||||
destinationDirectory.set(layout.buildDirectory.dir("libs"))
|
||||
archiveClassifier.set(DocsType.SOURCES)
|
||||
from(sourceSets.main.map { it.kotlin })
|
||||
from(provider {
|
||||
moduleSources.map {
|
||||
it.map { jarFile -> zipTree(jarFile) }
|
||||
}
|
||||
})
|
||||
}
|
||||
|
||||
buildPlugin {
|
||||
dependsOn(sourcesJar)
|
||||
from(sourcesJar) { into("lib/src") }
|
||||
}
|
||||
}
|
||||
|
||||
java {
|
||||
@@ -301,26 +244,10 @@ java {
|
||||
}
|
||||
|
||||
kotlin {
|
||||
explicitApi()
|
||||
jvmToolchain {
|
||||
languageVersion.set(JavaLanguageVersion.of(javaVersion))
|
||||
}
|
||||
|
||||
compilerOptions {
|
||||
jvmTarget.set(JvmTarget.fromTarget(javaVersion))
|
||||
|
||||
// See https://plugins.jetbrains.com/docs/intellij/using-kotlin.html#kotlin-standard-library
|
||||
// For the list of bundled versions
|
||||
apiVersion.set(KotlinVersion.KOTLIN_2_0)
|
||||
freeCompilerArgs = listOf(
|
||||
"-Xjvm-default=all-compatibility",
|
||||
|
||||
// Needed to compile the AceJump which uses kotlin beta
|
||||
// Without these two option compilation fails
|
||||
"-Xskip-prerelease-check",
|
||||
"-Xallow-unstable-dependencies",
|
||||
)
|
||||
// allWarningsAsErrors = true
|
||||
}
|
||||
}
|
||||
|
||||
gradle.projectsEvaluated {
|
||||
@@ -332,67 +259,134 @@ gradle.projectsEvaluated {
|
||||
|
||||
// --- Intellij plugin
|
||||
|
||||
intellijPlatform {
|
||||
pluginConfiguration {
|
||||
name = "IdeaVim"
|
||||
changeNotes.set(
|
||||
"""
|
||||
We’ve launched a program to reward quality contributions with a one-year All Products Pack subscription. Learn more at: <a href="https://github.com/JetBrains/ideavim/blob/master/CONTRIBUTING.md">CONTRIBUTING.md</a> .
|
||||
<br/>
|
||||
<br/>
|
||||
<a href="https://youtrack.jetbrains.com/issues/VIM?q=State:%20Fixed%20Fix%20versions:%20${version.get()}">Changelog</a>
|
||||
""".trimIndent()
|
||||
)
|
||||
intellij {
|
||||
version.set(ideaVersion)
|
||||
pluginName.set("IdeaVim")
|
||||
|
||||
ideaVersion {
|
||||
// Let the Gradle plugin set the since-build version. It defaults to the version of the IDE we're building against
|
||||
// specified as two components, `{branch}.{build}` (e.g., "241.15989"). There is no third component specified.
|
||||
// The until-build version defaults to `{branch}.*`, but we want to support _all_ future versions, so we set it
|
||||
// with a null provider (the provider is important).
|
||||
// By letting the Gradle plugin handle this, the Plugin DevKit IntelliJ plugin cannot help us with the "Usage of
|
||||
// IntelliJ API not available in older IDEs" inspection. However, since our since-build is the version we compile
|
||||
// against, we can never get an API that's newer - it would be an unresolved symbol.
|
||||
untilBuild.set(provider { null })
|
||||
}
|
||||
updateSinceUntilBuild.set(false)
|
||||
|
||||
downloadSources.set(downloadIdeaSources.toBoolean())
|
||||
instrumentCode.set(instrumentPluginCode.toBoolean())
|
||||
intellijRepository.set("https://www.jetbrains.com/intellij-repository")
|
||||
// Yaml is only used for testing. It's part of the IdeaIC distribution, but needs to be included as a reference
|
||||
plugins.set(listOf("java", "AceJump:3.8.11", "yaml"/*, "Pythonid:231.8109.2", "com.intellij.clion-swift:231.8109.4"*/))
|
||||
}
|
||||
|
||||
publishing {
|
||||
tasks {
|
||||
downloadRobotServerPlugin {
|
||||
version.set(remoteRobotVersion)
|
||||
}
|
||||
|
||||
publishPlugin {
|
||||
channels.set(publishChannels.split(","))
|
||||
token.set(publishToken)
|
||||
}
|
||||
|
||||
signing {
|
||||
signPlugin {
|
||||
certificateChain.set(providers.environmentVariable("CERTIFICATE_CHAIN"))
|
||||
privateKey.set(providers.environmentVariable("PRIVATE_KEY"))
|
||||
password.set(providers.environmentVariable("PRIVATE_KEY_PASSWORD"))
|
||||
}
|
||||
|
||||
pluginVerification {
|
||||
teamCityOutputFormat = true
|
||||
ides {
|
||||
recommended()
|
||||
runIdeForUiTests {
|
||||
systemProperty("robot-server.port", "8082")
|
||||
systemProperty("ide.mac.message.dialogs.as.sheets", "false")
|
||||
systemProperty("jb.privacy.policy.text", "<!--999.999-->")
|
||||
systemProperty("jb.consents.confirmation.enabled", "false")
|
||||
}
|
||||
|
||||
runPluginVerifier {
|
||||
downloadDir.set("${project.buildDir}/pluginVerifier/ides")
|
||||
teamCityOutputFormat.set(true)
|
||||
// ideVersions.set(listOf("IC-2021.3.4"))
|
||||
}
|
||||
|
||||
generateGrammarSource {
|
||||
maxHeapSize = "128m"
|
||||
arguments.addAll(listOf("-package", "com.maddyhome.idea.vim.vimscript.parser.generated", "-visitor"))
|
||||
outputDirectory = file("src/main/java/com/maddyhome/idea/vim/vimscript/parser/generated")
|
||||
}
|
||||
|
||||
named("compileKotlin") {
|
||||
dependsOn("generateGrammarSource")
|
||||
}
|
||||
named("compileTestKotlin") {
|
||||
dependsOn("generateTestGrammarSource")
|
||||
}
|
||||
|
||||
// Add plugin open API sources to the plugin ZIP
|
||||
val createOpenApiSourceJar by registering(Jar::class) {
|
||||
dependsOn("generateGrammarSource")
|
||||
// Java sources
|
||||
from(sourceSets.main.get().java) {
|
||||
include("**/com/maddyhome/idea/vim/**/*.java")
|
||||
}
|
||||
from(project(":vim-engine").sourceSets.main.get().java) {
|
||||
include("**/com/maddyhome/idea/vim/**/*.java")
|
||||
}
|
||||
// Kotlin sources
|
||||
from(kotlin.sourceSets.main.get().kotlin) {
|
||||
include("**/com/maddyhome/idea/vim/**/*.kt")
|
||||
}
|
||||
from(project(":vim-engine").kotlin.sourceSets.main.get().kotlin) {
|
||||
include("**/com/maddyhome/idea/vim/**/*.kt")
|
||||
}
|
||||
destinationDirectory.set(layout.buildDirectory.dir("libs"))
|
||||
archiveClassifier.set("src")
|
||||
}
|
||||
|
||||
buildPlugin {
|
||||
dependsOn(createOpenApiSourceJar)
|
||||
from(createOpenApiSourceJar) { into("lib/src") }
|
||||
}
|
||||
|
||||
val pluginVersion = version
|
||||
// Don't forget to update plugin.xml
|
||||
patchPluginXml {
|
||||
sinceBuild.set("231.7515.13")
|
||||
|
||||
// Get the latest available change notes from the changelog file
|
||||
changeNotes.set(
|
||||
provider {
|
||||
with(changelog) {
|
||||
val log = try {
|
||||
getUnreleased()
|
||||
} catch (e: MissingVersionException) {
|
||||
getOrNull(pluginVersion.toString()) ?: getLatest()
|
||||
}
|
||||
renderItem(
|
||||
log,
|
||||
org.jetbrains.changelog.Changelog.OutputType.HTML,
|
||||
)
|
||||
}
|
||||
},
|
||||
)
|
||||
}
|
||||
}
|
||||
|
||||
instrumentCode.set(instrumentPluginCode.toBoolean())
|
||||
// --- Linting
|
||||
|
||||
//ktlint {
|
||||
// version.set("0.48.2")
|
||||
//}
|
||||
|
||||
// --- Tests
|
||||
|
||||
tasks {
|
||||
test {
|
||||
useJUnitPlatform()
|
||||
exclude("**/propertybased/**")
|
||||
exclude("**/longrunning/**")
|
||||
exclude("/ui/**")
|
||||
}
|
||||
}
|
||||
|
||||
ksp {
|
||||
arg("generated_directory", "$projectDir/src/main/resources/ksp-generated")
|
||||
arg("vimscript_functions_file", "intellij_vimscript_functions.json")
|
||||
arg("ex_commands_file", "intellij_ex_commands.json")
|
||||
arg("commands_file", "intellij_commands.json")
|
||||
arg("extensions_file", "ideavim_extensions.json")
|
||||
tasks.register<Test>("testUi") {
|
||||
group = "verification"
|
||||
useJUnitPlatform()
|
||||
include("/ui/**")
|
||||
}
|
||||
|
||||
afterEvaluate {
|
||||
// tasks.named("kspKotlin").configure { dependsOn("clean") }
|
||||
tasks.named("kspTestFixturesKotlin").configure { enabled = false }
|
||||
tasks.named("kspTestFixturesKotlin").configure { enabled = false }
|
||||
tasks.named("kspTestKotlin").configure { enabled = false }
|
||||
}
|
||||
|
||||
|
||||
// --- Changelog
|
||||
|
||||
changelog {
|
||||
@@ -411,11 +405,20 @@ koverMerged {
|
||||
enable()
|
||||
}
|
||||
|
||||
kover {
|
||||
instrumentation {
|
||||
// set of test tasks names to exclude from instrumentation. The results of their execution will not be presented in the report
|
||||
excludeTasks += "testPropertyBased"
|
||||
excludeTasks += "testLongRunning"
|
||||
excludeTasks += "testWithNeovim"
|
||||
excludeTasks += "testUi"
|
||||
}
|
||||
}
|
||||
|
||||
// --- Slack notification
|
||||
|
||||
tasks.register<Task>("slackNotification") {
|
||||
tasks.register("slackNotification") {
|
||||
doLast {
|
||||
if (version.toString().last() != '0') return@doLast
|
||||
if (slackUrl.isBlank()) {
|
||||
println("Slack Url is not defined")
|
||||
return@doLast
|
||||
@@ -441,25 +444,20 @@ tasks.register<Task>("slackNotification") {
|
||||
""".trimIndent()
|
||||
|
||||
println("Parsed data: $slackDown")
|
||||
val post = URL(slackUrl)
|
||||
with(post.openConnection() as HttpURLConnection) {
|
||||
requestMethod = "POST"
|
||||
doOutput = true
|
||||
setRequestProperty("Content-Type", "application/json")
|
||||
|
||||
runBlocking {
|
||||
val client = HttpClient(CIO)
|
||||
try {
|
||||
val response = client.post(slackUrl) {
|
||||
contentType(ContentType.Application.Json)
|
||||
setBody(message)
|
||||
}
|
||||
outputStream.write(message.toByteArray())
|
||||
|
||||
val responseCode = response.status.value
|
||||
println("Response code: $responseCode")
|
||||
|
||||
val responseBody = response.body<String>()
|
||||
println(responseBody)
|
||||
} catch (e: Exception) {
|
||||
println("Error sending Slack notification: ${e.message}")
|
||||
throw e
|
||||
} finally {
|
||||
client.close()
|
||||
val postRc = responseCode
|
||||
println("Response code: $postRc")
|
||||
if (postRc == 200) {
|
||||
println(inputStream.bufferedReader().use { it.readText() })
|
||||
} else {
|
||||
println(errorStream.bufferedReader().use { it.readText() })
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -474,7 +472,7 @@ tasks.register<Task>("slackNotification") {
|
||||
// }
|
||||
|
||||
// --- Update authors
|
||||
tasks.register<Task>("updateAuthors") {
|
||||
tasks.register("updateAuthors") {
|
||||
doLast {
|
||||
val uncheckedEmails = setOf(
|
||||
"aleksei.plate@jetbrains.com",
|
||||
@@ -490,26 +488,24 @@ tasks.register<Task>("updateAuthors") {
|
||||
|
||||
val prId: String by project
|
||||
|
||||
tasks.register<Task>("updateMergedPr") {
|
||||
tasks.register("updateMergedPr") {
|
||||
doLast {
|
||||
val x = changelog.getUnreleased()
|
||||
println("x")
|
||||
// if (project.hasProperty("prId")) {
|
||||
// println("Got pr id: $prId")
|
||||
// updateMergedPr(prId.toInt())
|
||||
// } else {
|
||||
// error("Cannot get prId")
|
||||
// }
|
||||
if (project.hasProperty("prId")) {
|
||||
println("Got pr id: $prId")
|
||||
updateMergedPr(prId.toInt())
|
||||
} else {
|
||||
error("Cannot get prId")
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
tasks.register<Task>("updateChangelog") {
|
||||
tasks.register("updateChangelog") {
|
||||
doLast {
|
||||
updateChangelog()
|
||||
}
|
||||
}
|
||||
|
||||
tasks.register<Task>("updateYoutrackOnCommit") {
|
||||
tasks.register("updateYoutrackOnCommit") {
|
||||
doLast {
|
||||
updateYoutrackOnCommit()
|
||||
}
|
||||
@@ -520,22 +516,17 @@ val fixVersionsFieldId = "123-285"
|
||||
val fixVersionsFieldType = "VersionProjectCustomField"
|
||||
val fixVersionsElementType = "VersionBundleElement"
|
||||
|
||||
tasks.register<Task>("releaseActions") {
|
||||
tasks.register("releaseActions") {
|
||||
group = "other"
|
||||
doLast {
|
||||
if (releaseType == "patch") return@doLast
|
||||
|
||||
val tickets =
|
||||
getYoutrackTicketsByQuery("%23%7BReady+To+Release%7D%20and%20tag:%20%7BIdeaVim%20Released%20In%20EAP%7D%20")
|
||||
val tickets = getYoutrackTicketsByQuery("%23%7BReady+To+Release%7D")
|
||||
if (tickets.isNotEmpty()) {
|
||||
println("Updating statuses for tickets: $tickets")
|
||||
setYoutrackStatus(tickets, "Fixed")
|
||||
println("Checking if version $version exists...")
|
||||
val versionId = getVersionIdByName(version.toString())
|
||||
if (versionId == null) {
|
||||
if (getVersionIdByName(version.toString()) != null) {
|
||||
addReleaseToYoutrack(version.toString())
|
||||
} else {
|
||||
println("Version $version already exists in YouTrack. Version id: $versionId")
|
||||
println("Version $version is already exists in YouTrack")
|
||||
}
|
||||
setYoutrackFixVersion(tickets, version.toString())
|
||||
} else {
|
||||
@@ -544,7 +535,7 @@ tasks.register<Task>("releaseActions") {
|
||||
}
|
||||
}
|
||||
|
||||
tasks.register<Task>("integrationsTest") {
|
||||
tasks.register("integrationsTest") {
|
||||
group = "other"
|
||||
doLast {
|
||||
val testTicketId = "VIM-2784"
|
||||
@@ -593,7 +584,7 @@ fun guard(check: Boolean, ifWrong: () -> String) {
|
||||
}
|
||||
}
|
||||
|
||||
tasks.register<Task>("testUpdateChangelog") {
|
||||
tasks.register("testUpdateChangelog") {
|
||||
group = "verification"
|
||||
description = "This is a task to manually assert the correctness of the update tasks"
|
||||
doLast {
|
||||
@@ -614,8 +605,7 @@ fun addReleaseToYoutrack(name: String): String {
|
||||
println("Creating new release version in YouTrack: $name")
|
||||
|
||||
return runBlocking {
|
||||
val response =
|
||||
client.post("https://youtrack.jetbrains.com/api/admin/projects/$vimProjectId/customFields/$fixVersionsFieldId/bundle/values?fields=id,name") {
|
||||
val response = client.post("https://youtrack.jetbrains.com/api/admin/projects/$vimProjectId/customFields/$fixVersionsFieldId/bundle/values?fields=id,name") {
|
||||
contentType(ContentType.Application.Json)
|
||||
accept(ContentType.Application.Json)
|
||||
val request = buildJsonObject {
|
||||
@@ -632,8 +622,7 @@ fun getVersionIdByName(name: String): String? {
|
||||
val client = httpClient()
|
||||
|
||||
return runBlocking {
|
||||
val response =
|
||||
client.get("https://youtrack.jetbrains.com/api/admin/projects/$vimProjectId/customFields/$fixVersionsFieldId/bundle/values?fields=id,name&query=$name")
|
||||
val response = client.get("https://youtrack.jetbrains.com/api/admin/projects/$vimProjectId/customFields/$fixVersionsFieldId/bundle/values?fields=id,name&query=$name")
|
||||
response.body<JsonArray>().singleOrNull()?.jsonObject?.get("id")?.jsonPrimitive?.content
|
||||
}
|
||||
}
|
||||
@@ -671,8 +660,7 @@ fun setYoutrackStatus(tickets: Collection<String>, status: String) {
|
||||
runBlocking {
|
||||
for (ticket in tickets) {
|
||||
println("Try to set $ticket to $status")
|
||||
val response =
|
||||
client.post("https://youtrack.jetbrains.com/api/issues/$ticket?fields=customFields(id,name,value(id,name))") {
|
||||
val response = client.post("https://youtrack.jetbrains.com/api/issues/$ticket?fields=customFields(id,name,value(id,name))") {
|
||||
contentType(ContentType.Application.Json)
|
||||
accept(ContentType.Application.Json)
|
||||
val request = buildJsonObject {
|
||||
@@ -711,8 +699,7 @@ fun setYoutrackFixVersion(tickets: Collection<String>, version: String) {
|
||||
runBlocking {
|
||||
for (ticket in tickets) {
|
||||
println("Try to set fix version $version for $ticket")
|
||||
val response =
|
||||
client.post("https://youtrack.jetbrains.com/api/issues/$ticket?fields=customFields(id,name,value(id,name))") {
|
||||
val response = client.post("https://youtrack.jetbrains.com/api/issues/$ticket?fields=customFields(id,name,value(id,name))") {
|
||||
contentType(ContentType.Application.Json)
|
||||
accept(ContentType.Application.Json)
|
||||
val request = buildJsonObject {
|
||||
@@ -750,8 +737,7 @@ fun getYoutrackStatus(ticket: String): String {
|
||||
val client = httpClient()
|
||||
|
||||
return runBlocking {
|
||||
val response =
|
||||
client.get("https://youtrack.jetbrains.com/api/issues/$ticket/customFields/123-129?fields=value(name)")
|
||||
val response = client.get("https://youtrack.jetbrains.com/api/issues/$ticket/customFields/123-129?fields=value(name)")
|
||||
response.body<JsonObject>()["value"]!!.jsonObject.getValue("name").jsonPrimitive.content
|
||||
}
|
||||
}
|
||||
@@ -832,9 +818,7 @@ fun updateAuthors(uncheckedEmails: Set<String>) {
|
||||
org.intellij.markdown.parser.MarkdownParser(org.intellij.markdown.flavours.gfm.GFMFlavourDescriptor())
|
||||
val tree = parser.buildMarkdownTreeFromString(authors)
|
||||
|
||||
val contributorsSection = tree.children
|
||||
.filter { it is ListCompositeNode }
|
||||
.single { it.getTextInNode(authors).contains("yole") }
|
||||
val contributorsSection = tree.children[24]
|
||||
val existingEmails = mutableSetOf<String>()
|
||||
for (child in contributorsSection.children) {
|
||||
if (child.children.size > 1) {
|
||||
@@ -857,7 +841,7 @@ fun updateAuthors(uncheckedEmails: Set<String>) {
|
||||
}
|
||||
|
||||
fun List<Author>.toMdString(): String {
|
||||
return this.joinToString(separator = "") {
|
||||
return this.joinToString {
|
||||
"""
|
||||
|
|
||||
|* [![icon][mail]](mailto:${it.mail})
|
||||
@@ -987,12 +971,12 @@ fun changes(): List<Change> {
|
||||
println("Start changes processing")
|
||||
for (message in messages) {
|
||||
println("Processing '$message'...")
|
||||
val lowercaseMessage = message.lowercase()
|
||||
val lowercaseMessage = message.toLowerCase()
|
||||
val regex = "^fix\\((vim-\\d+)\\):".toRegex()
|
||||
val findResult = regex.find(lowercaseMessage)
|
||||
if (findResult != null) {
|
||||
println("Message matches")
|
||||
val value = findResult.groups[1]!!.value.uppercase()
|
||||
val value = findResult.groups[1]!!.value.toUpperCase()
|
||||
val shortMessage = message.drop(findResult.range.last + 1).trim()
|
||||
newFixes += Change(value, shortMessage)
|
||||
} else {
|
||||
|
@@ -1,6 +1,6 @@
|
||||
Welcome to the IdeaVim wiki!
|
||||
|
||||
- List of IdeaVim plugins: [plugins](IdeaVim%20Plugins.md)
|
||||
- Examples of `ideajoin` option (also known as "smart join"): ["ideajoin" examples](ideajoin-examples.md)
|
||||
- List of "set" commands: ["set" commands](set-commands.md)
|
||||
- Docs about "select" mode in vim: [select mode](Select-mode.md)
|
||||
- List of IdeaVim plugins: [[plugins|IdeaVim Plugins]]
|
||||
- Examples of `ideajoin` option (also known as "smart join"): [["ideajoin" examples|ideajoin-examples]]
|
||||
- List of "set" commands: [["set" commands|set-commands]]
|
||||
- Docs about "select" mode in vim: [[select mode|Select-mode]]
|
||||
|
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user