⁠
joseph burton: Add support for MixinExtras expressions (#2274 )
* Start on MixinExtras Expression language
* MEExpression color settings page
* MEExpression annotator
* MEExpression brace matcher and quote handler
* Switch LHS of MEExpression assignmentExpression to themselves be certain types of expression
* MEExpression language injection inside @Expression
* Fix formatting and licenses
* Add MIXINEXTRAS:EXPRESSION injection point and add @Expression annotation on completion
* Fix licenser errors
* Add new ME expression features
* Implement MixinExtras expression collect visitor
* Fix cast expressions
* Simple best-effort source matching for ME expressions
* Fix name expression source matching
* Fix MEName.isWildcard
* Fix MELitExpression source matching
* operationSign - operationTokenType
* Add built-in definitions
* Update MixinExtras
* Start with ME definition references
* Attempt to overhaul ME expression injection
* Some fixes to the new injection + navigation
* MixinExtras: Add handler signature support for expressions. (#2244 )
* Partially fix ME definition renaming
* Attempt to get inplace rename refactoring to work (it doesn't)
* MixinExtras: Use expression-suggested parameter names if they're present. (#2257 )
* Fix MEExpressionInjector. Rename refactoring works!
* Suppress deprecation warning
* ME expression `@Definition` find usages
* Fix/expressions int like types (#2261 )
* Mixin: Combine parameter and return type inspections.
* MixinExtras: Offer a choice between all valid int-like types.
* Mixin: Fix tests for handler signature inspection.
* Add simple keyword completion to ME expressions
* Why didn't my local ktlint tell me about these
* Store whether a declaration is a type in the ME PSI
* Add completions for items that already have a definition
* Extract some ME expression matching into its own class, and cache some more things
* Remove some debug...
* Start on MixinExtras Expression language
* MEExpression color settings page
* MEExpression annotator
* MEExpression brace matcher and quote handler
* Switch LHS of MEExpression assignmentExpression to themselves be certain types of expression
* MEExpression language injection inside @Expression
* Fix formatting and licenses
* Add MIXINEXTRAS:EXPRESSION injection point and add @Expression annotation on completion
* Fix licenser errors
* Add new ME expression features
* Implement MixinExtras expression collect visitor
* Fix cast expressions
* Simple best-effort source matching for ME expressions
* Fix name expression source matching
* Fix MEName.isWildcard
* Fix MELitExpression source matching
* operationSign - operationTokenType
* Add built-in definitions
* Update MixinExtras
* Start with ME definition references
* Attempt to overhaul ME expression injection
* Some fixes to the new injection + navigation
* MixinExtras: Add handler signature support for expressions. (#2244 )
* Partially fix ME definition renaming
* Attempt to get inplace rename refactoring to work (it doesn't)
* MixinExtras: Use expression-suggested parameter names if they're present. (#2257 )
* Fix MEExpressionInjector. Rename refactoring works!
* Suppress deprecation warning
* ME expression `@Definition` find usages
* Fix/expressions int like types (#2261 )
* Mixin: Combine parameter and return type inspections.
* MixinExtras: Offer a choice between all valid int-like types.
* Mixin: Fix tests for handler signature inspection.
* Add simple keyword completion to ME expressions
* Why didn't my local ktlint tell me about these
* Store whether a declaration is a type in the ME PSI
* Add completions for items that already have a definition
* Extract some ME expression matching into its own class, and cache some more things
* Remove some debug...
- /*
- * Minecraft Development for IntelliJ
- *
- * https://mcdev.io/
- *
- * Copyright (C) 2024 minecraft-dev
- *
- * This program is free software: you can redistribute it and/or modify
- * it under the terms of the GNU Lesser General Public License as published
- * by the Free Software Foundation, version 3.0 only.
- *
- * This program is distributed in the hope that it will be useful,
- * but WITHOUT ANY WARRANTY; without even the implied warranty of
- * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- * GNU General Public License for more details.
- *
- * You should have received a copy of the GNU Lesser General Public License
- * along with this program. If not, see <https://www.gnu.org/licenses/>.
- */
- package com.demonwav.mcdev.platform.mixin.expression.psi.mixins.impl
- import com.demonwav.mcdev.platform.mixin.expression.MESourceMatchContext
- import com.demonwav.mcdev.platform.mixin.expression.gen.psi.MEArguments
- import com.demonwav.mcdev.platform.mixin.expression.gen.psi.MEExpression
- import com.demonwav.mcdev.platform.mixin.expression.gen.psi.MEExpressionTypes
- import com.demonwav.mcdev.platform.mixin.expression.gen.psi.MEName
- import com.demonwav.mcdev.platform.mixin.expression.gen.psi.impl.MEExpressionImpl
- import com.demonwav.mcdev.platform.mixin.expression.meExpressionElementFactory
- import com.demonwav.mcdev.platform.mixin.expression.psi.mixins.MENewExpressionMixin
- import com.intellij.lang.ASTNode
- import com.intellij.psi.PsiArrayType
- import com.intellij.psi.PsiElement
- import com.intellij.psi.PsiNewExpression
- import com.intellij.psi.util.PsiUtil
- import com.intellij.psi.util.siblings
- abstract class MENewExpressionImplMixin(node: ASTNode) : MEExpressionImpl(node), MENewExpressionMixin {
- override val isArrayCreation get() = findChildByType<PsiElement>(MEExpressionTypes.TOKEN_LEFT_BRACKET) != null
- override val hasConstructorArguments get() = findChildByType<PsiElement>(MEExpressionTypes.TOKEN_LEFT_PAREN) != null
- override val dimensions get() = findChildrenByType<PsiElement>(MEExpressionTypes.TOKEN_LEFT_BRACKET).size
- override val dimExprTokens: List<MENewExpressionMixin.DimExprTokens> get() {
- val result = mutableListOf<MENewExpressionMixin.DimExprTokens>()
- var leftBracket: PsiElement? = findNotNullChildByType(MEExpressionTypes.TOKEN_LEFT_BRACKET)
- while (leftBracket != null) {
- var expr: MEExpression? = null
- var rightBracket: PsiElement? = null
- var nextLeftBracket: PsiElement? = null
- for (child in leftBracket.siblings(withSelf = false)) {
- if (child is MEExpression) {
- expr = child
- } else {
- when (child.node.elementType) {
- MEExpressionTypes.TOKEN_RIGHT_BRACKET -> rightBracket = child
- MEExpressionTypes.TOKEN_LEFT_BRACKET -> {
- nextLeftBracket = child
- break
- }
- }
- }
- }
- result += MENewExpressionMixin.DimExprTokens(leftBracket, expr, rightBracket)
- leftBracket = nextLeftBracket
- }
- return result
- }
- override val arrayInitializer get() = if (isArrayCreation) {
- arguments
- } else {
- null
- }
- override fun matchesJava(java: PsiElement, context: MESourceMatchContext): Boolean {
- if (java !is PsiNewExpression) {
- return false
- }
- if (isArrayCreation) {
- if (!java.isArrayCreation) {
- return false
- }
- val javaArrayType = java.type as? PsiArrayType ?: return false
- if (javaArrayType.arrayDimensions != dimensions) {
- return false
- }
- val matchesType = context.project.meExpressionElementFactory.createType(type)
- .matchesJava(javaArrayType.deepComponentType, context)
- if (!matchesType) {
- return false
- }
- val javaArrayDims = java.arrayDimensions
- val arrayDims = dimExprs
- if (javaArrayDims.size != arrayDims.size) {
- return false
- }
- if (!javaArrayDims.asSequence().zip(arrayDims.asSequence()).all { (javaArrayDim, arrayDim) ->
- val actualJavaDim = PsiUtil.skipParenthesizedExprDown(javaArrayDim) ?: return@all false
- arrayDim.matchesJava(actualJavaDim, context)
- }
- ) {
- return false
- }
- val javaArrayInitializer = java.arrayInitializer
- val arrayInitializer = this.arrayInitializer
- return if (javaArrayInitializer == null) {
- arrayInitializer == null
- } else {
- arrayInitializer?.matchesJava(javaArrayInitializer.initializers, context) == true
- }
- } else { // !isArrayCreation
- if (java.isArrayCreation) {
- return false
- }
- val javaType = java.type ?: return false
- val javaArgs = java.argumentList ?: return false
- return context.project.meExpressionElementFactory.createType(type).matchesJava(javaType, context) &&
- arguments?.matchesJava(javaArgs, context) == true
- }
- }
- override fun getInputExprs() = dimExprs + (arguments?.expressionList ?: emptyList())
- protected abstract val type: MEName
- protected abstract val dimExprs: List<MEExpression>
- protected abstract val arguments: MEArguments?
- }