mirror of
				https://github.com/chylex/IntelliJ-AceJump.git
				synced 2025-10-26 12:23:38 +01:00 
			
		
		
		
	Compare commits
	
		
			16 Commits
		
	
	
		
			broken-ide
			...
			568128f474
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 568128f474 | |||
| 018d29007c | |||
| cb814e0999 | |||
| 1fc06e8120 | |||
| 841f2fd125 | |||
| 6c8f19e311 | |||
| ce4f3b5f03 | |||
| 46f42c88eb | |||
| 43dfec940e | |||
| abe06ec7be | |||
| 3fc3cbc7f8 | |||
| 5979579042 | |||
| ea61d49aa6 | |||
| dbc6db108d | |||
| 01c38df82a | |||
| a3a86cf447 | 
| @@ -8,19 +8,22 @@ plugins { | |||||||
| } | } | ||||||
|  |  | ||||||
| group = "org.acejump" | group = "org.acejump" | ||||||
| version = "chylex-21" | version = "chylex-26" | ||||||
|  |  | ||||||
| repositories { | repositories { | ||||||
|   mavenCentral() |   mavenCentral() | ||||||
| } | } | ||||||
|  |  | ||||||
| intellij { | intellij { | ||||||
|   version.set("2024.1.4") |   version.set("2024.2") | ||||||
|   updateSinceUntilBuild.set(false) |   updateSinceUntilBuild.set(false) | ||||||
|   plugins.add("IdeaVIM:chylex-37") |    | ||||||
|  |   plugins.add("IdeaVIM:chylex-41") | ||||||
|  |   plugins.add("com.intellij.classic.ui:242.20224.159") | ||||||
|    |    | ||||||
|   pluginsRepositories { |   pluginsRepositories { | ||||||
|     custom("https://intellij.chylex.com") |     custom("https://intellij.chylex.com") | ||||||
|  |     marketplace() | ||||||
|   } |   } | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -33,7 +36,7 @@ dependencies { | |||||||
| } | } | ||||||
|  |  | ||||||
| tasks.patchPluginXml { | tasks.patchPluginXml { | ||||||
|   sinceBuild.set("241") |   sinceBuild.set("242") | ||||||
| } | } | ||||||
|  |  | ||||||
| tasks.buildSearchableOptions { | tasks.buildSearchableOptions { | ||||||
|   | |||||||
| @@ -1,5 +1,6 @@ | |||||||
| package org.acejump.action | package org.acejump.action | ||||||
|  |  | ||||||
|  | import com.intellij.openapi.actionSystem.ActionUpdateThread | ||||||
| import com.intellij.openapi.actionSystem.AnActionEvent | import com.intellij.openapi.actionSystem.AnActionEvent | ||||||
| import com.intellij.openapi.actionSystem.CommonDataKeys | import com.intellij.openapi.actionSystem.CommonDataKeys | ||||||
| import com.intellij.openapi.application.ApplicationManager | import com.intellij.openapi.application.ApplicationManager | ||||||
| @@ -15,7 +16,7 @@ import com.maddyhome.idea.vim.group.visual.vimSetSelection | |||||||
| import com.maddyhome.idea.vim.helper.inVisualMode | import com.maddyhome.idea.vim.helper.inVisualMode | ||||||
| import com.maddyhome.idea.vim.helper.vimSelectionStart | import com.maddyhome.idea.vim.helper.vimSelectionStart | ||||||
| import com.maddyhome.idea.vim.newapi.vim | import com.maddyhome.idea.vim.newapi.vim | ||||||
| import com.maddyhome.idea.vim.state.mode.Mode | import com.maddyhome.idea.vim.state.mode.Mode.OP_PENDING | ||||||
| import com.maddyhome.idea.vim.state.mode.SelectionType | import com.maddyhome.idea.vim.state.mode.SelectionType | ||||||
| import org.acejump.boundaries.StandardBoundaries.AFTER_CARET | import org.acejump.boundaries.StandardBoundaries.AFTER_CARET | ||||||
| import org.acejump.boundaries.StandardBoundaries.BEFORE_CARET | import org.acejump.boundaries.StandardBoundaries.BEFORE_CARET | ||||||
| @@ -51,8 +52,8 @@ sealed class AceVimAction : DumbAwareAction() { | |||||||
|           } |           } | ||||||
|           else { |           else { | ||||||
|             val vim = editor.vim |             val vim = editor.vim | ||||||
|  |             if (vim.mode is OP_PENDING) { | ||||||
|               val keyHandler = KeyHandler.getInstance() |               val keyHandler = KeyHandler.getInstance() | ||||||
|             if (keyHandler.isOperatorPending(vim.mode, keyHandler.keyHandlerState)) { |  | ||||||
|               val key = keyHandler.keyHandlerState.commandBuilder.keys.singleOrNull()?.keyChar |               val key = keyHandler.keyHandlerState.commandBuilder.keys.singleOrNull()?.keyChar | ||||||
|                |                | ||||||
|               keyHandler.fullReset(vim) |               keyHandler.fullReset(vim) | ||||||
| @@ -70,10 +71,10 @@ sealed class AceVimAction : DumbAwareAction() { | |||||||
|               if (action != null) { |               if (action != null) { | ||||||
|                 ApplicationManager.getApplication().invokeLater { |                 ApplicationManager.getApplication().invokeLater { | ||||||
|                   WriteAction.run<Nothing> { |                   WriteAction.run<Nothing> { | ||||||
|                     keyHandler.keyHandlerState.commandBuilder.pushCommandPart(action) |                     keyHandler.keyHandlerState.commandBuilder.addAction(action) | ||||||
|                      |                      | ||||||
|                     val cmd = keyHandler.keyHandlerState.commandBuilder.buildCommand() |                     val cmd = keyHandler.keyHandlerState.commandBuilder.buildCommand() | ||||||
|                     val operatorArguments = OperatorArguments(vim.mode is Mode.OP_PENDING, cmd.rawCount, injector.vimState.mode) |                     val operatorArguments = OperatorArguments(vim.mode is OP_PENDING, cmd.rawCount, injector.vimState.mode) | ||||||
|                      |                      | ||||||
|                     injector.vimState.executingCommand = cmd |                     injector.vimState.executingCommand = cmd | ||||||
|                     injector.actionExecutor.executeVimAction(vim, action, context, operatorArguments) |                     injector.actionExecutor.executeVimAction(vim, action, context, operatorArguments) | ||||||
| @@ -167,6 +168,10 @@ sealed class AceVimAction : DumbAwareAction() { | |||||||
|       action.presentation.isEnabled = action.getData(CommonDataKeys.EDITOR) != null |       action.presentation.isEnabled = action.getData(CommonDataKeys.EDITOR) != null | ||||||
|     } |     } | ||||||
|      |      | ||||||
|  |     override fun getActionUpdateThread(): ActionUpdateThread { | ||||||
|  |       return ActionUpdateThread.BGT | ||||||
|  |     } | ||||||
|  |      | ||||||
|     override fun actionPerformed(e: AnActionEvent) { |     override fun actionPerformed(e: AnActionEvent) { | ||||||
|       val editor = e.getData(CommonDataKeys.EDITOR) ?: return |       val editor = e.getData(CommonDataKeys.EDITOR) ?: return | ||||||
|       val session = SessionManager.start(editor, AceVimMode.JumpAllEditors.getJumpEditors(editor)) |       val session = SessionManager.start(editor, AceVimMode.JumpAllEditors.getJumpEditors(editor)) | ||||||
|   | |||||||
| @@ -18,6 +18,11 @@ sealed class EditorOffsetCache { | |||||||
|    */ |    */ | ||||||
|   abstract fun visibleArea(editor: Editor): Pair<Point, Point> |   abstract fun visibleArea(editor: Editor): Pair<Point, Point> | ||||||
|    |    | ||||||
|  |   /** | ||||||
|  |    * Returns whether the offset is in the visible area rectangle. | ||||||
|  |    */ | ||||||
|  |   abstract fun isVisible(editor: Editor, offset: Int): Boolean | ||||||
|  |    | ||||||
|   /** |   /** | ||||||
|    * Returns the editor offset at the provided pixel coordinate. |    * Returns the editor offset at the provided pixel coordinate. | ||||||
|    */ |    */ | ||||||
| @@ -36,6 +41,7 @@ sealed class EditorOffsetCache { | |||||||
|    |    | ||||||
|   private class Cache : EditorOffsetCache() { |   private class Cache : EditorOffsetCache() { | ||||||
|     private var visibleArea: Pair<Point, Point>? = null |     private var visibleArea: Pair<Point, Point>? = null | ||||||
|  |     private val lineToVisibleOffsetRange = Int2ObjectOpenHashMap<IntRange>() | ||||||
|     private val pointToOffset = Object2IntOpenHashMap<Point>().apply { defaultReturnValue(-1) } |     private val pointToOffset = Object2IntOpenHashMap<Point>().apply { defaultReturnValue(-1) } | ||||||
|     private val offsetToPoint = Int2ObjectOpenHashMap<Point>() |     private val offsetToPoint = Int2ObjectOpenHashMap<Point>() | ||||||
|      |      | ||||||
| @@ -43,6 +49,24 @@ sealed class EditorOffsetCache { | |||||||
|       return visibleArea ?: Uncached.visibleArea(editor).also { visibleArea = it } |       return visibleArea ?: Uncached.visibleArea(editor).also { visibleArea = it } | ||||||
|     } |     } | ||||||
|      |      | ||||||
|  |     override fun isVisible(editor: Editor, offset: Int): Boolean { | ||||||
|  |       val visualLine = editor.offsetToVisualLine(offset, false) | ||||||
|  |        | ||||||
|  |       var visibleRange = lineToVisibleOffsetRange.get(visualLine) | ||||||
|  |       if (visibleRange == null) { | ||||||
|  |         val (topLeft, bottomRight) = visibleArea(editor) | ||||||
|  |         val lineY = editor.visualLineToY(visualLine) | ||||||
|  |          | ||||||
|  |         val firstVisibleOffset = xyToOffset(editor, Point(topLeft.x, lineY)) | ||||||
|  |         val lastVisibleOffset = xyToOffset(editor, Point(bottomRight.x, lineY)) | ||||||
|  |          | ||||||
|  |         visibleRange = firstVisibleOffset..lastVisibleOffset | ||||||
|  |         lineToVisibleOffsetRange.put(visualLine, visibleRange) | ||||||
|  |       } | ||||||
|  |        | ||||||
|  |       return offset in visibleRange | ||||||
|  |     } | ||||||
|  |      | ||||||
|     override fun xyToOffset(editor: Editor, pos: Point): Int { |     override fun xyToOffset(editor: Editor, pos: Point): Int { | ||||||
|       val offset = pointToOffset.getInt(pos) |       val offset = pointToOffset.getInt(pos) | ||||||
|        |        | ||||||
| @@ -51,7 +75,6 @@ sealed class EditorOffsetCache { | |||||||
|       } |       } | ||||||
|        |        | ||||||
|       return Uncached.xyToOffset(editor, pos).also { |       return Uncached.xyToOffset(editor, pos).also { | ||||||
|         @Suppress("ReplacePutWithAssignment") |  | ||||||
|         pointToOffset.put(pos, it) |         pointToOffset.put(pos, it) | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
| @@ -64,7 +87,6 @@ sealed class EditorOffsetCache { | |||||||
|       } |       } | ||||||
|        |        | ||||||
|       return Uncached.offsetToXY(editor, offset).also { |       return Uncached.offsetToXY(editor, offset).also { | ||||||
|         @Suppress("ReplacePutWithAssignment") |  | ||||||
|         offsetToPoint.put(offset, it) |         offsetToPoint.put(offset, it) | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
| @@ -80,6 +102,15 @@ sealed class EditorOffsetCache { | |||||||
|       ) |       ) | ||||||
|     } |     } | ||||||
|      |      | ||||||
|  |     override fun isVisible(editor: Editor, offset: Int): Boolean { | ||||||
|  |       val (topLeft, bottomRight) = visibleArea(editor) | ||||||
|  |       val pos = offsetToXY(editor, offset) | ||||||
|  |       val x = pos.x | ||||||
|  |       val y = pos.y | ||||||
|  |        | ||||||
|  |       return x >= topLeft.x && y >= topLeft.y && x <= bottomRight.x && y <= bottomRight.y | ||||||
|  |     } | ||||||
|  |      | ||||||
|     override fun xyToOffset(editor: Editor, pos: Point): Int { |     override fun xyToOffset(editor: Editor, pos: Point): Int { | ||||||
|       return editor.logicalPositionToOffset(editor.xyToLogicalPosition(pos)) |       return editor.logicalPositionToOffset(editor.xyToLogicalPosition(pos)) | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -13,23 +13,7 @@ enum class StandardBoundaries : Boundaries { | |||||||
|     } |     } | ||||||
|      |      | ||||||
|     override fun isOffsetInside(editor: Editor, offset: Int, cache: EditorOffsetCache): Boolean { |     override fun isOffsetInside(editor: Editor, offset: Int, cache: EditorOffsetCache): Boolean { | ||||||
|        |       return cache.isVisible(editor, offset) | ||||||
|       // If we are not using a cache, calling getOffsetRange will cause additional 1-2 pixel coordinate -> offset lookups, which is a lot |  | ||||||
|       // more expensive than one lookup compared against the visible area. |  | ||||||
|        |  | ||||||
|       // However, if we are using a cache, it's likely that the topmost and bottommost positions are already cached whereas the provided |  | ||||||
|       // offset isn't, so we save a lookup for every offset outside the range. |  | ||||||
|        |  | ||||||
|       if (cache !== EditorOffsetCache.Uncached && offset !in getOffsetRange(editor, cache)) { |  | ||||||
|         return false |  | ||||||
|       } |  | ||||||
|        |  | ||||||
|       val (topLeft, bottomRight) = cache.visibleArea(editor) |  | ||||||
|       val pos = cache.offsetToXY(editor, offset) |  | ||||||
|       val x = pos.x |  | ||||||
|       val y = pos.y |  | ||||||
|        |  | ||||||
|       return x >= topLeft.x && y >= topLeft.y && x <= bottomRight.x && y <= bottomRight.y |  | ||||||
|     } |     } | ||||||
|   }, |   }, | ||||||
|    |    | ||||||
|   | |||||||
| @@ -20,6 +20,7 @@ class AceConfig : PersistentStateComponent<AceSettings> { | |||||||
|      |      | ||||||
|     val layout get() = settings.layout |     val layout get() = settings.layout | ||||||
|     val minQueryLength get() = settings.minQueryLength |     val minQueryLength get() = settings.minQueryLength | ||||||
|  |     val invertUppercaseMode get() = settings.invertUppercaseMode | ||||||
|     val editorFadeOpacity get() = settings.editorFadeOpacity |     val editorFadeOpacity get() = settings.editorFadeOpacity | ||||||
|     val jumpModeColor get() = settings.jumpModeColor |     val jumpModeColor get() = settings.jumpModeColor | ||||||
|     val tagForegroundColor1 get() = settings.tagForegroundColor1 |     val tagForegroundColor1 get() = settings.tagForegroundColor1 | ||||||
|   | |||||||
| @@ -13,9 +13,9 @@ class AceConfigurable : Configurable { | |||||||
|    |    | ||||||
|   override fun isModified() = |   override fun isModified() = | ||||||
|     panel.allowedChars != settings.allowedChars || |     panel.allowedChars != settings.allowedChars || | ||||||
|       panel.prefixChars != settings.prefixChars || |  | ||||||
|       panel.keyboardLayout != settings.layout || |       panel.keyboardLayout != settings.layout || | ||||||
|       panel.minQueryLengthInt != settings.minQueryLength || |       panel.minQueryLengthInt != settings.minQueryLength || | ||||||
|  |       panel.invertUppercaseMode != settings.invertUppercaseMode || | ||||||
|       panel.editorFadeOpacityPercent != settings.editorFadeOpacity || |       panel.editorFadeOpacityPercent != settings.editorFadeOpacity || | ||||||
|       panel.jumpModeColor != settings.jumpModeColor || |       panel.jumpModeColor != settings.jumpModeColor || | ||||||
|       panel.tagForegroundColor1 != settings.tagForegroundColor1 || |       panel.tagForegroundColor1 != settings.tagForegroundColor1 || | ||||||
| @@ -24,9 +24,9 @@ class AceConfigurable : Configurable { | |||||||
|    |    | ||||||
|   override fun apply() { |   override fun apply() { | ||||||
|     settings.allowedChars = panel.allowedChars |     settings.allowedChars = panel.allowedChars | ||||||
|     settings.prefixChars = panel.prefixChars |  | ||||||
|     settings.layout = panel.keyboardLayout |     settings.layout = panel.keyboardLayout | ||||||
|     settings.minQueryLength = panel.minQueryLengthInt ?: settings.minQueryLength |     settings.minQueryLength = panel.minQueryLengthInt ?: settings.minQueryLength | ||||||
|  |     settings.invertUppercaseMode = panel.invertUppercaseMode | ||||||
|     settings.editorFadeOpacity = panel.editorFadeOpacityPercent |     settings.editorFadeOpacity = panel.editorFadeOpacityPercent | ||||||
|     panel.jumpModeColor?.let { settings.jumpModeColor = it } |     panel.jumpModeColor?.let { settings.jumpModeColor = it } | ||||||
|     panel.tagForegroundColor1?.let { settings.tagForegroundColor1 = it } |     panel.tagForegroundColor1?.let { settings.tagForegroundColor1 = it } | ||||||
|   | |||||||
| @@ -8,7 +8,7 @@ import java.awt.Color | |||||||
| data class AceSettings( | data class AceSettings( | ||||||
|   var layout: KeyLayout = QWERTY, |   var layout: KeyLayout = QWERTY, | ||||||
|   var allowedChars: String = layout.allChars, |   var allowedChars: String = layout.allChars, | ||||||
|   var prefixChars: String = ";", |   var invertUppercaseMode: Boolean = false, | ||||||
|   var minQueryLength: Int = 1, |   var minQueryLength: Int = 1, | ||||||
|   var editorFadeOpacity: Int = 70, |   var editorFadeOpacity: Int = 70, | ||||||
|    |    | ||||||
|   | |||||||
| @@ -2,19 +2,20 @@ package org.acejump.config | |||||||
|  |  | ||||||
| import com.intellij.openapi.ui.ComboBox | import com.intellij.openapi.ui.ComboBox | ||||||
| import com.intellij.ui.ColorPanel | import com.intellij.ui.ColorPanel | ||||||
|  | import com.intellij.ui.components.JBCheckBox | ||||||
| import com.intellij.ui.components.JBSlider | import com.intellij.ui.components.JBSlider | ||||||
| import com.intellij.ui.components.JBTextArea | import com.intellij.ui.components.JBTextArea | ||||||
| import com.intellij.ui.components.JBTextField | import com.intellij.ui.components.JBTextField | ||||||
| import com.intellij.ui.layout.Cell | import com.intellij.ui.dsl.builder.COLUMNS_LARGE | ||||||
| import com.intellij.ui.layout.GrowPolicy.MEDIUM_TEXT | import com.intellij.ui.dsl.builder.COLUMNS_SHORT | ||||||
| import com.intellij.ui.layout.GrowPolicy.SHORT_TEXT | import com.intellij.ui.dsl.builder.columns | ||||||
| import com.intellij.ui.layout.panel | import com.intellij.ui.dsl.builder.panel | ||||||
| import org.acejump.input.KeyLayout | import org.acejump.input.KeyLayout | ||||||
| import java.awt.Color | import java.awt.Color | ||||||
|  | import java.awt.Dimension | ||||||
| import java.awt.Font | import java.awt.Font | ||||||
| import java.util.Hashtable | import java.util.Hashtable | ||||||
| import javax.swing.JCheckBox | import javax.swing.JCheckBox | ||||||
| import javax.swing.JComponent |  | ||||||
| import javax.swing.JLabel | import javax.swing.JLabel | ||||||
| import javax.swing.JPanel | import javax.swing.JPanel | ||||||
| import javax.swing.JSlider | import javax.swing.JSlider | ||||||
| @@ -27,10 +28,10 @@ import kotlin.reflect.KProperty | |||||||
| @Suppress("UsePropertyAccessSyntax") | @Suppress("UsePropertyAccessSyntax") | ||||||
| internal class AceSettingsPanel { | internal class AceSettingsPanel { | ||||||
|   private val tagAllowedCharsField = JBTextField() |   private val tagAllowedCharsField = JBTextField() | ||||||
|   private val tagPrefixCharsField = JBTextField() |  | ||||||
|   private val keyboardLayoutCombo = ComboBox<KeyLayout>() |   private val keyboardLayoutCombo = ComboBox<KeyLayout>() | ||||||
|   private val keyboardLayoutArea = JBTextArea().apply { isEditable = false } |   private val keyboardLayoutArea = JBTextArea().apply { isEditable = false } | ||||||
|   private val minQueryLengthField = JBTextField() |   private val minQueryLengthField = JBTextField() | ||||||
|  |   private val invertUppercaseModeCheckBox = JBCheckBox("Invert uppercase mode") | ||||||
|   private val editorFadeOpacitySlider = JBSlider(0, 10).apply { |   private val editorFadeOpacitySlider = JBSlider(0, 10).apply { | ||||||
|     labelTable = Hashtable((0..10).associateWith { JLabel("${it * 10}") }) |     labelTable = Hashtable((0..10).associateWith { JLabel("${it * 10}") }) | ||||||
|     paintTrack = true |     paintTrack = true | ||||||
| @@ -38,6 +39,7 @@ internal class AceSettingsPanel { | |||||||
|     paintTicks = true |     paintTicks = true | ||||||
|     minorTickSpacing = 1 |     minorTickSpacing = 1 | ||||||
|     majorTickSpacing = 1 |     majorTickSpacing = 1 | ||||||
|  |     minimumSize = Dimension(275, minimumSize.height) | ||||||
|   } |   } | ||||||
|   private val jumpModeColorWheel = ColorPanel() |   private val jumpModeColorWheel = ColorPanel() | ||||||
|   private val tagForeground1ColorWheel = ColorPanel() |   private val tagForeground1ColorWheel = ColorPanel() | ||||||
| @@ -46,55 +48,45 @@ internal class AceSettingsPanel { | |||||||
|    |    | ||||||
|   init { |   init { | ||||||
|     tagAllowedCharsField.apply { font = Font("monospaced", font.style, font.size) } |     tagAllowedCharsField.apply { font = Font("monospaced", font.style, font.size) } | ||||||
|     tagPrefixCharsField.apply { font = Font("monospaced", font.style, font.size) } |  | ||||||
|     keyboardLayoutArea.apply { font = Font("monospaced", font.style, font.size) } |     keyboardLayoutArea.apply { font = Font("monospaced", font.style, font.size) } | ||||||
|     keyboardLayoutCombo.setupEnumItems { keyChars = it.rows.joinToString("\n") } |     keyboardLayoutCombo.setupEnumItems { keyChars = it.rows.joinToString("\n") } | ||||||
|   } |   } | ||||||
|    |    | ||||||
|   internal val rootPanel: JPanel = panel { |   internal val rootPanel: JPanel = panel { | ||||||
|     fun Cell.short(component: JComponent) = component(growPolicy = SHORT_TEXT) |     group("Characters and Layout") { | ||||||
|     fun Cell.medium(component: JComponent) = component(growPolicy = MEDIUM_TEXT) |       row("Allowed characters in tags:") { cell(tagAllowedCharsField).columns(COLUMNS_LARGE) } | ||||||
|      |       row("Keyboard layout:") { cell(keyboardLayoutCombo).columns(COLUMNS_SHORT) } | ||||||
|     titledRow("Characters and Layout") { |       row("Keyboard design:") { cell(keyboardLayoutArea).columns(COLUMNS_SHORT) } | ||||||
|       row("Allowed characters in tags:") { medium(tagAllowedCharsField) } |  | ||||||
|       row("Allowed prefix characters in tags:") { medium(tagPrefixCharsField) } |  | ||||||
|       row("Keyboard layout:") { short(keyboardLayoutCombo) } |  | ||||||
|       row("Keyboard design:") { short(keyboardLayoutArea) } |  | ||||||
|     } |     } | ||||||
|      |      | ||||||
|     titledRow("Behavior") { |     group("Behavior") { | ||||||
|       row("Minimum typed characters (1-10):") { short(minQueryLengthField) } |       row("Minimum typed characters (1-10):") { cell(minQueryLengthField).columns(COLUMNS_SHORT) } | ||||||
|  |       row { cell(invertUppercaseModeCheckBox) } | ||||||
|     } |     } | ||||||
|      |      | ||||||
|     titledRow("Colors") { |     group("Colors") { | ||||||
|       row("Caret background:") { |       row("Caret background:") { | ||||||
|         cell { |         cell(jumpModeColorWheel) | ||||||
|           component(jumpModeColorWheel) |  | ||||||
|         } |  | ||||||
|       } |       } | ||||||
|       row("Tag foreground:") { |       row("Tag foreground:") { | ||||||
|         cell { |         cell(tagForeground1ColorWheel) | ||||||
|           component(tagForeground1ColorWheel) |         cell(tagForeground2ColorWheel) | ||||||
|           component(tagForeground2ColorWheel) |  | ||||||
|         } |  | ||||||
|       } |       } | ||||||
|       row("Search highlight:") { |       row("Search highlight:") { | ||||||
|         cell { |         cell(searchHighlightColorWheel) | ||||||
|           component(searchHighlightColorWheel) |  | ||||||
|         } |  | ||||||
|       } |       } | ||||||
|       row("Editor fade opacity (%):") { |       row("Editor fade opacity (%):") { | ||||||
|         medium(editorFadeOpacitySlider) |         cell(editorFadeOpacitySlider) | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|    |    | ||||||
|   // Property-to-property delegation: https://stackoverflow.com/q/45074596/1772342 |   // Property-to-property delegation: https://stackoverflow.com/q/45074596/1772342 | ||||||
|   internal var allowedChars by tagAllowedCharsField |   internal var allowedChars by tagAllowedCharsField | ||||||
|   internal var prefixChars by tagPrefixCharsField |  | ||||||
|   internal var keyboardLayout by keyboardLayoutCombo |   internal var keyboardLayout by keyboardLayoutCombo | ||||||
|   internal var keyChars by keyboardLayoutArea |   internal var keyChars by keyboardLayoutArea | ||||||
|   internal var minQueryLength by minQueryLengthField |   internal var minQueryLength by minQueryLengthField | ||||||
|  |   internal var invertUppercaseMode by invertUppercaseModeCheckBox | ||||||
|   internal var editorFadeOpacity by editorFadeOpacitySlider |   internal var editorFadeOpacity by editorFadeOpacitySlider | ||||||
|   internal var jumpModeColor by jumpModeColorWheel |   internal var jumpModeColor by jumpModeColorWheel | ||||||
|   internal var tagForegroundColor1 by tagForeground1ColorWheel |   internal var tagForegroundColor1 by tagForeground1ColorWheel | ||||||
| @@ -111,9 +103,9 @@ internal class AceSettingsPanel { | |||||||
|    |    | ||||||
|   fun reset(settings: AceSettings) { |   fun reset(settings: AceSettings) { | ||||||
|     allowedChars = settings.allowedChars |     allowedChars = settings.allowedChars | ||||||
|     prefixChars = settings.prefixChars |  | ||||||
|     keyboardLayout = settings.layout |     keyboardLayout = settings.layout | ||||||
|     minQueryLength = settings.minQueryLength.toString() |     minQueryLength = settings.minQueryLength.toString() | ||||||
|  |     invertUppercaseMode = settings.invertUppercaseMode | ||||||
|     editorFadeOpacityPercent = settings.editorFadeOpacity |     editorFadeOpacityPercent = settings.editorFadeOpacity | ||||||
|     jumpModeColor = settings.jumpModeColor |     jumpModeColor = settings.jumpModeColor | ||||||
|     tagForegroundColor1 = settings.tagForegroundColor1 |     tagForegroundColor1 = settings.tagForegroundColor1 | ||||||
| @@ -123,7 +115,7 @@ internal class AceSettingsPanel { | |||||||
|    |    | ||||||
|   // Removal pending support for https://youtrack.jetbrains.com/issue/KT-8575 |   // Removal pending support for https://youtrack.jetbrains.com/issue/KT-8575 | ||||||
|    |    | ||||||
|   private operator fun JTextComponent.getValue(a: AceSettingsPanel, p: KProperty<*>) = text.toLowerCase() |   private operator fun JTextComponent.getValue(a: AceSettingsPanel, p: KProperty<*>) = text.lowercase() | ||||||
|   private operator fun JTextComponent.setValue(a: AceSettingsPanel, p: KProperty<*>, s: String) = setText(s) |   private operator fun JTextComponent.setValue(a: AceSettingsPanel, p: KProperty<*>, s: String) = setText(s) | ||||||
|    |    | ||||||
|   private operator fun ColorPanel.getValue(a: AceSettingsPanel, p: KProperty<*>) = selectedColor |   private operator fun ColorPanel.getValue(a: AceSettingsPanel, p: KProperty<*>) = selectedColor | ||||||
|   | |||||||
| @@ -10,7 +10,6 @@ import com.intellij.openapi.editor.actionSystem.TypedActionHandler | |||||||
|  * sessions' own handlers. |  * sessions' own handlers. | ||||||
|  */ |  */ | ||||||
| internal object EditorKeyListener : TypedActionHandler { | internal object EditorKeyListener : TypedActionHandler { | ||||||
|   private val action = TypedAction.getInstance() |  | ||||||
|   private val attached = mutableMapOf<Editor, TypedActionHandler>() |   private val attached = mutableMapOf<Editor, TypedActionHandler>() | ||||||
|   private var originalHandler: TypedActionHandler? = null |   private var originalHandler: TypedActionHandler? = null | ||||||
|    |    | ||||||
| @@ -20,8 +19,9 @@ internal object EditorKeyListener : TypedActionHandler { | |||||||
|    |    | ||||||
|   fun attach(editor: Editor, callback: TypedActionHandler) { |   fun attach(editor: Editor, callback: TypedActionHandler) { | ||||||
|     if (attached.isEmpty()) { |     if (attached.isEmpty()) { | ||||||
|       originalHandler = action.rawHandler |       val typedAction = TypedAction.getInstance() | ||||||
|       action.setupRawHandler(this) |       originalHandler = typedAction.rawHandler | ||||||
|  |       typedAction.setupRawHandler(this) | ||||||
|     } |     } | ||||||
|      |      | ||||||
|     attached[editor] = callback |     attached[editor] = callback | ||||||
| @@ -31,7 +31,7 @@ internal object EditorKeyListener : TypedActionHandler { | |||||||
|     attached.remove(editor) |     attached.remove(editor) | ||||||
|      |      | ||||||
|     if (attached.isEmpty()) { |     if (attached.isEmpty()) { | ||||||
|       originalHandler?.let(action::setupRawHandler) |       originalHandler?.let(TypedAction.getInstance()::setupRawHandler) | ||||||
|       originalHandler = null |       originalHandler = null | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|   | |||||||
| @@ -5,13 +5,18 @@ package org.acejump.input | |||||||
|  * ergonomically difficult they are to press. |  * ergonomically difficult they are to press. | ||||||
|  */ |  */ | ||||||
| @Suppress("unused", "SpellCheckingInspection") | @Suppress("unused", "SpellCheckingInspection") | ||||||
| enum class KeyLayout(internal val rows: Array<String>, priority: String, internal val characterRemapping: Map<Char, Char> = emptyMap()) { | enum class KeyLayout( | ||||||
|  |   internal val rows: Array<String>, | ||||||
|  |   priority: String, | ||||||
|  |   private val characterSides: Pair<Set<Char>, Set<Char>> = Pair(emptySet(), emptySet()), | ||||||
|  |   internal val characterRemapping: Map<Char, Char> = emptyMap(), | ||||||
|  | ) { | ||||||
|   COLEMK(arrayOf("1234567890", "qwfpgjluy", "arstdhneio", "zxcvbkm"), priority = "tndhseriaovkcmbxzgjplfuwyq5849673210"), |   COLEMK(arrayOf("1234567890", "qwfpgjluy", "arstdhneio", "zxcvbkm"), priority = "tndhseriaovkcmbxzgjplfuwyq5849673210"), | ||||||
|   WORKMN(arrayOf("1234567890", "qdrwbjfup", "ashtgyneoi", "zxmcvkl"), priority = "tnhegysoaiclvkmxzwfrubjdpq5849673210"), |   WORKMN(arrayOf("1234567890", "qdrwbjfup", "ashtgyneoi", "zxmcvkl"), priority = "tnhegysoaiclvkmxzwfrubjdpq5849673210"), | ||||||
|   DVORAK(arrayOf("1234567890", "pyfgcrl", "aoeuidhtns", "qjkxbmwvz"), priority = "uhetidonasxkbjmqwvzgfycprl5849673210"), |   DVORAK(arrayOf("1234567890", "pyfgcrl", "aoeuidhtns", "qjkxbmwvz"), priority = "uhetidonasxkbjmqwvzgfycprl5849673210"), | ||||||
|   QWERTY(arrayOf("1234567890", "qwertyuiop", "asdfghjkl", "zxcvbnm"), priority = "fjghdkslavncmbxzrutyeiwoqp5849673210"), |   QWERTY(arrayOf("1234567890", "qwertyuiop", "asdfghjkl", "zxcvbnm"), priority = "fjghdkslavncmbxzrutyeiwoqp5849673210", characterSides = sides(listOf("123456", "qwert", "asdfg", "zxcvb"), listOf("7890", "yuiop", "hjkl", "nm"))), | ||||||
|   QWERTZ(arrayOf("1234567890", "qwertzuiop", "asdfghjkl", "yxcvbnm"), priority = "fjghdkslavncmbxyrutzeiwoqp5849673210"), |   QWERTZ(arrayOf("1234567890", "qwertzuiop", "asdfghjkl", "yxcvbnm"), priority = "fjghdkslavncmbxyrutzeiwoqp5849673210", characterSides = sides(listOf("123456", "qwert", "asdfg", "yxcvb"), listOf("7890", "zuiop", "hjkl", "nm"))), | ||||||
|   QWERTZ_CZ(arrayOf("1234567890", "qwertzuiop", "asdfghjkl", "yxcvbnm"), priority = "fjghdkslavncmbxyrutzeiwoqp5849673210", characterRemapping =  mapOf( |   QWERTZ_CZ(arrayOf("1234567890", "qwertzuiop", "asdfghjkl", "yxcvbnm"), priority = "fjghdkslavncmbxyrutzeiwoqp5849673210", characterSides = sides(listOf("123456", "qwert", "asdfg", "yxcvb"), listOf("7890", "zuiop", "hjkl", "nm")), characterRemapping = mapOf( | ||||||
|     '+' to '1', |     '+' to '1', | ||||||
|     'ě' to '2', |     'ě' to '2', | ||||||
|     'š' to '3', |     'š' to '3', | ||||||
| @@ -28,9 +33,20 @@ enum class KeyLayout(internal val rows: Array<String>, priority: String, interna | |||||||
|   NORMAN(arrayOf("1234567890", "qwdfkjurl", "asetgynioh", "zxcvbpm"), priority = "tneigysoahbvpcmxzjkufrdlwq5849673210"); |   NORMAN(arrayOf("1234567890", "qwdfkjurl", "asetgynioh", "zxcvbpm"), priority = "tneigysoahbvpcmxzjkufrdlwq5849673210"); | ||||||
|    |    | ||||||
|   internal val allChars = rows.joinToString("").toCharArray().apply(CharArray::sort).joinToString("") |   internal val allChars = rows.joinToString("").toCharArray().apply(CharArray::sort).joinToString("") | ||||||
|   internal val allPriorities = priority.mapIndexed { index, char -> char to index }.toMap() |   private val allPriorities = priority.mapIndexed { index, char -> char to index }.toMap() | ||||||
|    |    | ||||||
|   internal inline fun priority(crossinline tagToChar: (String) -> Char): (String) -> Int { |   fun priority(char: Char): Int { | ||||||
|     return { allPriorities.getOrDefault(tagToChar(it), Int.MAX_VALUE) } |     return allPriorities[char] ?: allChars.length | ||||||
|  |   } | ||||||
|  |    | ||||||
|  |   fun areOnSameSide(c1: Char, c2: Char): Boolean { | ||||||
|  |     return (c1 in characterSides.first && c2 in characterSides.first) || (c1 in characterSides.second && c2 in characterSides.second) | ||||||
|   } |   } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | private fun sides(left: List<String>, right: List<String>): Pair<Set<Char>, Set<Char>> { | ||||||
|  |   return Pair( | ||||||
|  |     left.flatMapTo(mutableSetOf()) { it.toCharArray().toSet() }, | ||||||
|  |     right.flatMapTo(mutableSetOf()) { it.toCharArray().toSet() } | ||||||
|  |   ) | ||||||
|  | } | ||||||
|   | |||||||
| @@ -1,23 +1,22 @@ | |||||||
| package org.acejump.input | package org.acejump.input | ||||||
|  |  | ||||||
| import org.acejump.config.AceSettings | import org.acejump.config.AceSettings | ||||||
|  | import kotlin.math.pow | ||||||
|  | import kotlin.math.roundToInt | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * Stores data specific to the selected keyboard layout. We want to assign tags with easily reachable keys first, and ideally have tags |  * Stores data specific to the selected keyboard layout. We want to assign tags with easily reachable keys first, and ideally have tags | ||||||
|  * with repeated keys (ex. FF, JJ) or adjacent keys (ex. GH, UJ). |  * with repeated keys (ex. FF, JJ) or adjacent keys (ex. GH, UJ). | ||||||
|  */ |  */ | ||||||
| internal object KeyLayoutCache { | internal object KeyLayoutCache { | ||||||
|   /** |   lateinit var allowedTagsSorted: List<String> | ||||||
|    * Returns all possible two key tags, pre-sorted according to [tagOrder]. |  | ||||||
|    */ |  | ||||||
|   lateinit var allPossibleTagsLowercase: List<String> |  | ||||||
|     private set |     private set | ||||||
|    |    | ||||||
|   /** |   /** | ||||||
|    * Called before any lazily initialized properties are used, to ensure that they are initialized even if the settings are missing. |    * Called before any lazily initialized properties are used, to ensure that they are initialized even if the settings are missing. | ||||||
|    */ |    */ | ||||||
|   fun ensureInitialized(settings: AceSettings) { |   fun ensureInitialized(settings: AceSettings) { | ||||||
|     if (!::allPossibleTagsLowercase.isInitialized) { |     if (!::allowedTagsSorted.isInitialized) { | ||||||
|       reset(settings) |       reset(settings) | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
| @@ -26,22 +25,38 @@ internal object KeyLayoutCache { | |||||||
|    * Re-initializes cached data according to updated settings. |    * Re-initializes cached data according to updated settings. | ||||||
|    */ |    */ | ||||||
|   fun reset(settings: AceSettings) { |   fun reset(settings: AceSettings) { | ||||||
|     @Suppress("ConvertLambdaToReference") |     val allowedChars = processCharList(settings.allowedChars).ifEmpty { processCharList(settings.layout.allChars) } | ||||||
|     val allSuffixChars = processCharList(settings.allowedChars).ifEmpty { processCharList(settings.layout.allChars).toList() } |     val allowedTags = mutableSetOf<String>() | ||||||
|     val allPrefixChars = processCharList(settings.prefixChars).filterNot(allSuffixChars::contains).plus("") |  | ||||||
|      |      | ||||||
|     val tagOrder = compareBy( |     for (c1 in allowedChars) { | ||||||
|       String::length, |       allowedTags.add("$c1") | ||||||
|       { if (it.length == 1) Int.MIN_VALUE else allPrefixChars.indexOf(it.first().toString()) }, |  | ||||||
|       settings.layout.priority(String::last) |  | ||||||
|     ) |  | ||||||
|        |        | ||||||
|     allPossibleTagsLowercase = allSuffixChars |       for (c2 in allowedChars) { | ||||||
|       .flatMap { suffix -> allPrefixChars.map { prefix -> "$prefix$suffix" } } |         if (c1 != c2) { | ||||||
|       .sortedWith(tagOrder) |           allowedTags.add("$c1$c2") | ||||||
|   } |         } | ||||||
|    |       } | ||||||
|   private fun processCharList(charList: String): Set<String> { |     } | ||||||
|     return charList.toCharArray().map(Char::lowercase).toSet() |      | ||||||
|  |     allowedTagsSorted = allowedTags.sortedBy { rankPriority(settings.layout, it) } | ||||||
|  |   } | ||||||
|  |    | ||||||
|  |   private fun processCharList(charList: String): List<Char> { | ||||||
|  |     return charList.toCharArray().map(Char::lowercaseChar).distinct() | ||||||
|  |   } | ||||||
|  |    | ||||||
|  |   private fun rankPriority(layout: KeyLayout, tag: String): Int { | ||||||
|  |     val c1 = tag.first() | ||||||
|  |     val p1 = (1.0 + layout.priority(c1)).pow(3) | ||||||
|  |      | ||||||
|  |     if (tag.length == 1) { | ||||||
|  |       return p1.roundToInt() | ||||||
|  |     } | ||||||
|  |      | ||||||
|  |     val c2 = tag.last() | ||||||
|  |     val p2 = (1.0 + layout.priority(c2)).pow(3) | ||||||
|  |      | ||||||
|  |     val multiplier = if (layout.areOnSameSide(c1, c2)) 2 else 1 | ||||||
|  |     return (((p1 * 50) + p2 + 1000) * multiplier).roundToInt() | ||||||
|   } |   } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -3,6 +3,7 @@ package org.acejump.search | |||||||
| import com.intellij.openapi.editor.Editor | import com.intellij.openapi.editor.Editor | ||||||
| import it.unimi.dsi.fastutil.ints.IntArrayList | import it.unimi.dsi.fastutil.ints.IntArrayList | ||||||
| import org.acejump.boundaries.Boundaries | import org.acejump.boundaries.Boundaries | ||||||
|  | import org.acejump.boundaries.EditorOffsetCache | ||||||
| import org.acejump.clone | import org.acejump.clone | ||||||
| import org.acejump.config.AceConfig | import org.acejump.config.AceConfig | ||||||
| import org.acejump.immutableText | import org.acejump.immutableText | ||||||
| @@ -11,15 +12,16 @@ import org.acejump.matchesAt | |||||||
| /** | /** | ||||||
|  * Searches editor text for matches of a [SearchQuery], and updates previous results when the user [refineQuery]s a character. |  * Searches editor text for matches of a [SearchQuery], and updates previous results when the user [refineQuery]s a character. | ||||||
|  */ |  */ | ||||||
| class SearchProcessor private constructor(query: SearchQuery, val boundaries: Boundaries, private val results: MutableMap<Editor, IntArrayList>) { | class SearchProcessor private constructor(query: SearchQuery, val boundaries: Boundaries, val invertUppercaseMode: Boolean, private val results: MutableMap<Editor, IntArrayList>) { | ||||||
|   internal constructor(editors: List<Editor>, query: SearchQuery, boundaries: Boundaries) : this(query, boundaries, mutableMapOf()) { |   internal constructor(editors: List<Editor>, query: SearchQuery, boundaries: Boundaries, invertUppercaseMode: Boolean) : this(query, boundaries, invertUppercaseMode, mutableMapOf()) { | ||||||
|     val regex = query.toRegex() |     val regex = query.toRegex(invertUppercaseMode) | ||||||
|      |      | ||||||
|     if (regex != null) { |     if (regex != null) { | ||||||
|       for (editor in editors) { |       for (editor in editors) { | ||||||
|  |         val cache = EditorOffsetCache.new() | ||||||
|         val offsets = IntArrayList() |         val offsets = IntArrayList() | ||||||
|          |          | ||||||
|         val offsetRange = boundaries.getOffsetRange(editor) |         val offsetRange = boundaries.getOffsetRange(editor, cache) | ||||||
|         var result = regex.find(editor.immutableText, offsetRange.first) |         var result = regex.find(editor.immutableText, offsetRange.first) | ||||||
|          |          | ||||||
|         while (result != null) { |         while (result != null) { | ||||||
| @@ -29,7 +31,7 @@ class SearchProcessor private constructor(query: SearchQuery, val boundaries: Bo | |||||||
|           if (highlightEnd > offsetRange.last) { |           if (highlightEnd > offsetRange.last) { | ||||||
|             break |             break | ||||||
|           } |           } | ||||||
|           else if (boundaries.isOffsetInside(editor, index) && !editor.foldingModel.isOffsetCollapsed(index)) { |           else if (boundaries.isOffsetInside(editor, index, cache) && !editor.foldingModel.isOffsetCollapsed(index)) { | ||||||
|             offsets.add(index) |             offsets.add(index) | ||||||
|           } |           } | ||||||
|            |            | ||||||
|   | |||||||
| @@ -21,42 +21,37 @@ internal sealed class SearchQuery { | |||||||
|   /** |   /** | ||||||
|    * Converts the query into a regular expression to find the initial matches. |    * Converts the query into a regular expression to find the initial matches. | ||||||
|    */ |    */ | ||||||
|   abstract fun toRegex(): Regex? |   abstract fun toRegex(invertUppercaseMode: Boolean): Regex? | ||||||
|    |    | ||||||
|   /** |   /** | ||||||
|    * Searches for all occurrences of a literal text query. |    * Searches for all occurrences of a literal text query. | ||||||
|    * If the first character of the query is lowercase, then the entire query will be case-insensitive, |    * If the first character of the query is lowercase, then the entire query will be case-insensitive, | ||||||
|    * and only beginnings of words and camel humps will be matched. |    * and only beginnings of words and camel humps will be matched. | ||||||
|    */ |    */ | ||||||
|   class Literal(override val rawText: String, val excludeMiddlesOfWords: Boolean) : SearchQuery() { |   class Literal(override val rawText: String) : SearchQuery() { | ||||||
|     init { |     init { | ||||||
|       require(rawText.isNotEmpty()) |       require(rawText.isNotEmpty()) | ||||||
|     } |     } | ||||||
|      |      | ||||||
|     override fun refine(char: Char): SearchQuery { |     override fun refine(char: Char): SearchQuery { | ||||||
|       return Literal(rawText + char, excludeMiddlesOfWords) |       return Literal(rawText + char) | ||||||
|     } |     } | ||||||
|      |      | ||||||
|     override fun getHighlightLength(text: CharSequence, offset: Int): Int { |     override fun getHighlightLength(text: CharSequence, offset: Int): Int { | ||||||
|       return text.countMatchingCharacters(offset, rawText) |       return text.countMatchingCharacters(offset, rawText) | ||||||
|     } |     } | ||||||
|      |      | ||||||
|     override fun toRegex(): Regex { |     override fun toRegex(invertUppercaseMode: Boolean): Regex { | ||||||
|       val firstChar = rawText.first() |       val firstChar = rawText.first() | ||||||
|       val pattern = if (firstChar.isLowerCase()) { |       val pattern = if (firstChar.isLowerCase() xor invertUppercaseMode) { | ||||||
|         if (excludeMiddlesOfWords) { |  | ||||||
|           val firstCharUppercasePattern = Regex.escape(firstChar.uppercaseChar().toString()) |  | ||||||
|           val firstCharPattern = Regex.escape(firstChar.toString()) |  | ||||||
|           val remainingPattern = if (rawText.length > 1) Regex.escape(rawText.drop(1)) else "" |  | ||||||
|           "(?:$firstCharUppercasePattern|(?<![a-zA-Z])$firstCharPattern)$remainingPattern" |  | ||||||
|         } |  | ||||||
|         else { |  | ||||||
|         val fullPattern = Regex.escape(rawText) |         val fullPattern = Regex.escape(rawText) | ||||||
|         "(?i)$fullPattern" |         "(?i)$fullPattern" | ||||||
|       } |       } | ||||||
|       } |  | ||||||
|       else { |       else { | ||||||
|         Regex.escape(rawText) |         val firstCharUppercasePattern = Regex.escape(firstChar.uppercase()) | ||||||
|  |         val firstCharLowercasePattern = Regex.escape(firstChar.lowercase()) | ||||||
|  |         val remainingPattern = if (rawText.length > 1) Regex.escape(rawText.drop(1)) else "" | ||||||
|  |         "(?:$firstCharUppercasePattern|(?<![a-zA-Z])$firstCharLowercasePattern)$remainingPattern" | ||||||
|       } |       } | ||||||
|        |        | ||||||
|       return Regex(pattern, setOf(RegexOption.MULTILINE)) |       return Regex(pattern, setOf(RegexOption.MULTILINE)) | ||||||
| @@ -70,14 +65,14 @@ internal sealed class SearchQuery { | |||||||
|     override val rawText = "" |     override val rawText = "" | ||||||
|      |      | ||||||
|     override fun refine(char: Char): SearchQuery { |     override fun refine(char: Char): SearchQuery { | ||||||
|       return Literal(char.toString(), excludeMiddlesOfWords = false) |       return Literal(char.toString()) | ||||||
|     } |     } | ||||||
|      |      | ||||||
|     override fun getHighlightLength(text: CharSequence, offset: Int): Int { |     override fun getHighlightLength(text: CharSequence, offset: Int): Int { | ||||||
|       return 1 |       return 1 | ||||||
|     } |     } | ||||||
|      |      | ||||||
|     override fun toRegex(): Regex { |     override fun toRegex(invertUppercaseMode: Boolean): Regex { | ||||||
|       return Regex(pattern, setOf(RegexOption.MULTILINE, RegexOption.IGNORE_CASE)) |       return Regex(pattern, setOf(RegexOption.MULTILINE, RegexOption.IGNORE_CASE)) | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|   | |||||||
| @@ -3,6 +3,7 @@ package org.acejump.search | |||||||
| import com.google.common.collect.ArrayListMultimap | import com.google.common.collect.ArrayListMultimap | ||||||
| import com.intellij.openapi.editor.Editor | import com.intellij.openapi.editor.Editor | ||||||
| import it.unimi.dsi.fastutil.ints.IntList | import it.unimi.dsi.fastutil.ints.IntList | ||||||
|  | import it.unimi.dsi.fastutil.objects.Object2IntOpenHashMap | ||||||
| import org.acejump.boundaries.EditorOffsetCache | import org.acejump.boundaries.EditorOffsetCache | ||||||
| import org.acejump.boundaries.StandardBoundaries.VISIBLE_ON_SCREEN | import org.acejump.boundaries.StandardBoundaries.VISIBLE_ON_SCREEN | ||||||
| import org.acejump.input.KeyLayoutCache | import org.acejump.input.KeyLayoutCache | ||||||
| @@ -40,7 +41,7 @@ class Tagger(private val editors: List<Editor>, results: Map<Editor, IntList>) { | |||||||
|       .flatMap { (editor, sites) -> sites.map { site -> Tag(editor, site) } } |       .flatMap { (editor, sites) -> sites.map { site -> Tag(editor, site) } } | ||||||
|       .sortedWith(siteOrder(editors, caches)) |       .sortedWith(siteOrder(editors, caches)) | ||||||
|      |      | ||||||
|     tagMap = KeyLayoutCache.allPossibleTagsLowercase.zip(tagSites).toMap() |     tagMap = generateTags(tagSites.size).zip(tagSites).toMap() | ||||||
|   } |   } | ||||||
|    |    | ||||||
|   internal fun type(char: Char): TaggingResult { |   internal fun type(char: Char): TaggingResult { | ||||||
| @@ -61,6 +62,68 @@ class Tagger(private val editors: List<Editor>, results: Map<Editor, IntList>) { | |||||||
|   } |   } | ||||||
|    |    | ||||||
|   private companion object { |   private companion object { | ||||||
|  |     private fun generateTags(tagCount: Int): List<String> { | ||||||
|  |       val allowedTagsSorted = KeyLayoutCache.allowedTagsSorted | ||||||
|  |       val tags = mutableListOf<String>() | ||||||
|  |        | ||||||
|  |       val containedSingleCharTags = mutableSetOf<Char>() | ||||||
|  |       val blockedSingleCharTags = mutableSetOf<Char>() | ||||||
|  |       val doubleCharTagCountsByFirstChar = Object2IntOpenHashMap<Char>() | ||||||
|  |        | ||||||
|  |       for (tag in allowedTagsSorted) { | ||||||
|  |         val firstChar = tag.first() | ||||||
|  |          | ||||||
|  |         if (tag.length == 1) { | ||||||
|  |           if (firstChar in blockedSingleCharTags) { | ||||||
|  |             continue | ||||||
|  |           } | ||||||
|  |            | ||||||
|  |           containedSingleCharTags.add(firstChar) | ||||||
|  |         } | ||||||
|  |         else { | ||||||
|  |           if (containedSingleCharTags.remove(firstChar)) { | ||||||
|  |             tags.remove(firstChar.toString()) | ||||||
|  |           } | ||||||
|  |            | ||||||
|  |           blockedSingleCharTags.add(firstChar) | ||||||
|  |           doubleCharTagCountsByFirstChar.addTo(firstChar, 1) | ||||||
|  |         } | ||||||
|  |          | ||||||
|  |         tags.add(tag) | ||||||
|  |          | ||||||
|  |         if (tags.size >= tagCount) { | ||||||
|  |           break | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |        | ||||||
|  |       // In rare cases, the final tag list may contain a double character tag that is the only tag starting with its first character, | ||||||
|  |       // so we replace it with the single character tag. | ||||||
|  |       for (entry in doubleCharTagCountsByFirstChar.object2IntEntrySet()) { | ||||||
|  |         if (entry.intValue != 1) { | ||||||
|  |           continue | ||||||
|  |         } | ||||||
|  |          | ||||||
|  |         tags.removeAt(tags.indexOfFirst { it.first() == entry.key }) | ||||||
|  |          | ||||||
|  |         val tag = entry.key.toString() | ||||||
|  |         var previousTagIndex = -1 | ||||||
|  |          | ||||||
|  |         // The implementation of searching where to place the single character tag is theoretically slow, | ||||||
|  |         // but getting here is so rare it doesn't matter. | ||||||
|  |         for (i in allowedTagsSorted.indexOf(tag) - 1 downTo 0) { | ||||||
|  |           previousTagIndex = tags.indexOf(allowedTagsSorted[i]) | ||||||
|  |            | ||||||
|  |           if (previousTagIndex != -1) { | ||||||
|  |             break | ||||||
|  |           } | ||||||
|  |         } | ||||||
|  |          | ||||||
|  |         tags.add(previousTagIndex + 1, tag) | ||||||
|  |       } | ||||||
|  |        | ||||||
|  |       return tags | ||||||
|  |     } | ||||||
|  |      | ||||||
|     private fun sortResults(results: Map<Editor, IntList>, caches: Map<Editor, EditorOffsetCache>) { |     private fun sortResults(results: Map<Editor, IntList>, caches: Map<Editor, EditorOffsetCache>) { | ||||||
|       for ((editor, offsets) in results) { |       for ((editor, offsets) in results) { | ||||||
|         val cache = caches.getValue(editor) |         val cache = caches.getValue(editor) | ||||||
| @@ -99,12 +162,11 @@ class Tagger(private val editors: List<Editor>, results: Map<Editor, IntList>) { | |||||||
|         return@Comparator if (aIsVisible) -1 else 1 |         return@Comparator if (aIsVisible) -1 else 1 | ||||||
|       } |       } | ||||||
|        |        | ||||||
|       val aPosition = aCaches.offsetToXY(aEditor, a.offset) |       val firstEditor = editorPriority[0] | ||||||
|       val bPosition = bCaches.offsetToXY(bEditor, b.offset) |       val caretPosition = caches.getValue(firstEditor).offsetToXY(firstEditor, firstEditor.caretModel.offset) | ||||||
|        |        | ||||||
|       val caretPosition = editorPriority[0].offsetToXY(editorPriority[0].caretModel.offset) |       val aDistance = aCaches.offsetToXY(aEditor, a.offset).distanceSq(caretPosition) | ||||||
|       val aDistance = aPosition.distanceSq(caretPosition) |       val bDistance = bCaches.offsetToXY(bEditor, b.offset).distanceSq(caretPosition) | ||||||
|       val bDistance = bPosition.distanceSq(caretPosition) |  | ||||||
|        |        | ||||||
|       return@Comparator aDistance.compareTo(bDistance) |       return@Comparator aDistance.compareTo(bDistance) | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -113,7 +113,7 @@ class Session(private val mainEditor: Editor, private val jumpEditors: List<Edit | |||||||
|     } |     } | ||||||
|      |      | ||||||
|     setMode(mode()) |     setMode(mode()) | ||||||
|     state = SessionState.WaitForKey(actions, jumpEditors, defaultBoundary) |     state = SessionState.WaitForKey(actions, jumpEditors, defaultBoundary, AceConfig.invertUppercaseMode) | ||||||
|   } |   } | ||||||
|    |    | ||||||
|   /** |   /** | ||||||
| @@ -128,7 +128,7 @@ class Session(private val mainEditor: Editor, private val jumpEditors: List<Edit | |||||||
|       canvas.setMarkers(emptyList()) |       canvas.setMarkers(emptyList()) | ||||||
|     } |     } | ||||||
|      |      | ||||||
|     val processor = SearchProcessor(jumpEditors, SearchQuery.RegularExpression(pattern.regex), defaultBoundary) |     val processor = SearchProcessor(jumpEditors, SearchQuery.RegularExpression(pattern.regex), defaultBoundary, AceConfig.invertUppercaseMode) | ||||||
|     textHighlighter.renderOccurrences(processor.resultsCopy, processor.query) |     textHighlighter.renderOccurrences(processor.resultsCopy, processor.query) | ||||||
|      |      | ||||||
|     state = SessionState.SelectTag(actions, jumpEditors, processor) |     state = SessionState.SelectTag(actions, jumpEditors, processor) | ||||||
|   | |||||||
| @@ -2,6 +2,7 @@ package org.acejump.session | |||||||
|  |  | ||||||
| import com.intellij.openapi.editor.Editor | import com.intellij.openapi.editor.Editor | ||||||
| import org.acejump.boundaries.Boundaries | import org.acejump.boundaries.Boundaries | ||||||
|  | import org.acejump.boundaries.StandardBoundaries | ||||||
| import org.acejump.config.AceConfig | import org.acejump.config.AceConfig | ||||||
| import org.acejump.search.SearchProcessor | import org.acejump.search.SearchProcessor | ||||||
| import org.acejump.search.SearchQuery | import org.acejump.search.SearchQuery | ||||||
| @@ -15,9 +16,10 @@ sealed interface SessionState { | |||||||
|     private val actions: SessionActions, |     private val actions: SessionActions, | ||||||
|     private val jumpEditors: List<Editor>, |     private val jumpEditors: List<Editor>, | ||||||
|     private val defaultBoundary: Boundaries, |     private val defaultBoundary: Boundaries, | ||||||
|  |     private val invertUppercaseMode: Boolean, | ||||||
|   ) : SessionState { |   ) : SessionState { | ||||||
|     override fun type(char: Char): TypeResult { |     override fun type(char: Char): TypeResult { | ||||||
|       val searchProcessor = SearchProcessor(jumpEditors, SearchQuery.Literal(char.toString(), excludeMiddlesOfWords = true), defaultBoundary) |       val searchProcessor = SearchProcessor(jumpEditors, SearchQuery.Literal(char.toString()), defaultBoundary, invertUppercaseMode) | ||||||
|        |        | ||||||
|       return if (searchProcessor.isQueryFinished) { |       return if (searchProcessor.isQueryFinished) { | ||||||
|         TypeResult.ChangeState(SelectTag(actions, jumpEditors, searchProcessor)) |         TypeResult.ChangeState(SelectTag(actions, jumpEditors, searchProcessor)) | ||||||
| @@ -63,12 +65,22 @@ sealed interface SessionState { | |||||||
|     override fun type(char: Char): TypeResult { |     override fun type(char: Char): TypeResult { | ||||||
|       if (char == ' ') { |       if (char == ' ') { | ||||||
|         val query = searchProcessor.query |         val query = searchProcessor.query | ||||||
|         if (query is SearchQuery.Literal && query.excludeMiddlesOfWords) { |         if (query is SearchQuery.Literal) { | ||||||
|           val newQuery = SearchQuery.Literal(query.rawText, excludeMiddlesOfWords = false) |           val newBoundaries = when (searchProcessor.boundaries) { | ||||||
|           val newSearchProcessor = SearchProcessor(jumpEditors, newQuery, searchProcessor.boundaries) |             StandardBoundaries.VISIBLE_ON_SCREEN -> StandardBoundaries.AFTER_CARET | ||||||
|  |             StandardBoundaries.AFTER_CARET       -> StandardBoundaries.BEFORE_CARET | ||||||
|  |             StandardBoundaries.BEFORE_CARET      -> StandardBoundaries.VISIBLE_ON_SCREEN | ||||||
|  |             else                                 -> searchProcessor.boundaries | ||||||
|  |           } | ||||||
|  |            | ||||||
|  |           val newSearchProcessor = SearchProcessor(jumpEditors, query, newBoundaries, searchProcessor.invertUppercaseMode) | ||||||
|           return TypeResult.ChangeState(SelectTag(actions, jumpEditors, newSearchProcessor)) |           return TypeResult.ChangeState(SelectTag(actions, jumpEditors, newSearchProcessor)) | ||||||
|         } |         } | ||||||
|       } |       } | ||||||
|  |       else if (char == '\n') { | ||||||
|  |         val newSearchProcessor = SearchProcessor(jumpEditors, searchProcessor.query, searchProcessor.boundaries, !searchProcessor.invertUppercaseMode) | ||||||
|  |         return TypeResult.ChangeState(SelectTag(actions, jumpEditors, newSearchProcessor)) | ||||||
|  |       } | ||||||
|        |        | ||||||
|       return when (val result = tagger.type(AceConfig.layout.characterRemapping.getOrDefault(char, char))) { |       return when (val result = tagger.type(AceConfig.layout.characterRemapping.getOrDefault(char, char))) { | ||||||
|         is TaggingResult.Nothing -> TypeResult.Nothing |         is TaggingResult.Nothing -> TypeResult.Nothing | ||||||
|   | |||||||
| @@ -12,10 +12,13 @@ import java.awt.Rectangle | |||||||
|  * Describes a 1 or 2 character shortcut that points to a specific character in the editor. |  * Describes a 1 or 2 character shortcut that points to a specific character in the editor. | ||||||
|  */ |  */ | ||||||
| internal class TagMarker( | internal class TagMarker( | ||||||
|   private val tag: CharArray, |   private val firstChar: String, | ||||||
|  |   private val secondChar: String, | ||||||
|   val offset: Int |   val offset: Int | ||||||
| ) { | ) { | ||||||
|   private val length = tag.size |   private constructor(tag: String, offset: Int) : this(tag.first().toString(), tag.drop(1), offset) | ||||||
|  |    | ||||||
|  |   private val length = firstChar.length + secondChar.length | ||||||
|    |    | ||||||
|   companion object { |   companion object { | ||||||
|     /** |     /** | ||||||
| @@ -28,7 +31,7 @@ internal class TagMarker( | |||||||
|      * character ([typedTag]) matches the first [tag] character, only the second [tag] character is displayed. |      * character ([typedTag]) matches the first [tag] character, only the second [tag] character is displayed. | ||||||
|      */ |      */ | ||||||
|     fun create(tag: String, offset: Int, typedTag: String): TagMarker { |     fun create(tag: String, offset: Int, typedTag: String): TagMarker { | ||||||
|       return TagMarker(tag.drop(typedTag.length).toCharArray(), offset) |       return TagMarker(tag.drop(typedTag.length), offset) | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|    |    | ||||||
| @@ -65,11 +68,11 @@ internal class TagMarker( | |||||||
|      |      | ||||||
|     g.font = font.tagFont |     g.font = font.tagFont | ||||||
|     g.color = font.foregroundColor1 |     g.color = font.foregroundColor1 | ||||||
|     g.drawChars(tag, 0, 1, x, y) |     g.drawString(firstChar, x, y) | ||||||
|      |      | ||||||
|     if (tag.size > 1) { |     if (secondChar.isNotEmpty()) { | ||||||
|       g.color = font.foregroundColor2 |       g.color = font.foregroundColor2 | ||||||
|       g.drawChars(tag, 1, length - 1, x + font.tagCharWidth, y) |       g.drawString(secondChar, x + font.tagCharWidth, y) | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|    |    | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user