⁠
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.MEExpression
- import com.demonwav.mcdev.platform.mixin.expression.gen.psi.MEExpressionTypes
- import com.demonwav.mcdev.platform.mixin.expression.gen.psi.impl.MEExpressionImpl
- import com.demonwav.mcdev.platform.mixin.expression.psi.mixins.MELitExpressionMixin
- import com.intellij.lang.ASTNode
- import com.intellij.psi.JavaTokenType
- import com.intellij.psi.PsiElement
- import com.intellij.psi.PsiLiteral
- import com.intellij.psi.PsiUnaryExpression
- import com.intellij.psi.util.PsiUtil
- import com.intellij.util.IncorrectOperationException
- abstract class MELitExpressionImplMixin(node: ASTNode) : MEExpressionImpl(node), MELitExpressionMixin {
- override val value get() = when (node.firstChildNode.elementType) {
- MEExpressionTypes.TOKEN_NULL_LIT -> null
- MEExpressionTypes.TOKEN_MINUS -> {
- when (node.lastChildNode.elementType) {
- MEExpressionTypes.TOKEN_INT_LIT -> {
- val text = node.lastChildNode.text
- if (text.startsWith("0x")) {
- "-${text.substring(2)}".toLongOrNull(16)
- } else {
- "-$text".toLongOrNull()
- }
- }
- MEExpressionTypes.TOKEN_DEC_LIT -> {
- "-${node.lastChildNode.text}".toDoubleOrNull()
- }
- else -> throw IncorrectOperationException("Invalid number literal format")
- }
- }
- MEExpressionTypes.TOKEN_BOOL_LIT -> node.chars[0] == 't'
- MEExpressionTypes.TOKEN_INT_LIT -> {
- val text = this.text
- if (text.startsWith("0x")) {
- text.substring(2).toLongOrNull(16)
- } else {
- text.toLongOrNull()
- }
- }
- MEExpressionTypes.TOKEN_DEC_LIT -> text.toDoubleOrNull()
- else -> {
- val text = this.text
- if (text.length >= 2) {
- text.substring(1, text.length - 1).replace("\\'", "'").replace("\\\\", "\\")
- } else {
- null
- }
- }
- }
- override val isNull get() = node.firstChildNode.elementType == MEExpressionTypes.TOKEN_NULL_LIT
- override val isString get() = node.firstChildNode.elementType == MEExpressionTypes.TOKEN_STRING_TERMINATOR
- override val minusToken get() = node.firstChildNode.takeIf { it.elementType == MEExpressionTypes.TOKEN_MINUS }
- override fun matchesJava(java: PsiElement, context: MESourceMatchContext): Boolean {
- return when (java) {
- is PsiLiteral -> {
- val value = this.value
- val javaValue = java.value.widened
- // MixinExtras compares floats as strings
- when (value) {
- is Double -> javaValue is Double && value.toString() == javaValue.toString()
- is String -> {
- val matchesChar =
- value.length == 1 && javaValue is Long && value.firstOrNull()?.code?.toLong() == javaValue
- matchesChar || value == javaValue
- }
- else -> value == javaValue
- }
- }
- is PsiUnaryExpression -> {
- if (java.operationTokenType != JavaTokenType.MINUS) {
- return false
- }
- val javaOperand = PsiUtil.skipParenthesizedExprDown(java.operand) ?: return false
- if (javaOperand !is PsiLiteral) {
- return false
- }
- val value = this.value
- val javaValue = javaOperand.value.widened
- when (value) {
- is Long -> javaValue == -value
- is Double -> javaValue is Double && javaValue.toString() == (-value).toString()
- else -> false
- }
- }
- else -> false
- }
- }
- override fun getInputExprs() = emptyList<MEExpression>()
- private val Any?.widened: Any? get() = when (this) {
- is Int -> toLong()
- is Float -> toDouble()
- is Char -> code.toLong()
- else -> this
- }
- }