KT-4820: Added new tests, introduced functionality to detect overridden functions, other minor fixes
This commit is contained in:
+1
-1
@@ -1,3 +1,3 @@
|
||||
for ((i,x) in foo.withIndices()) {
|
||||
for ((i, x) in foo.withIndices()) {
|
||||
|
||||
}
|
||||
@@ -23,18 +23,13 @@ import org.jetbrains.jet.lang.psi.JetDotQualifiedExpression
|
||||
import com.intellij.psi.util.PsiTreeUtil
|
||||
import com.intellij.psi.PsiDocumentManager
|
||||
import org.jetbrains.jet.plugin.project.AnalyzerFacadeWithCache
|
||||
import org.jetbrains.jet.di.InjectorForMacros
|
||||
import org.jetbrains.jet.lang.resolve.BindingContext
|
||||
import org.jetbrains.jet.lang.types.TypeUtils
|
||||
import org.jetbrains.jet.lang.resolve.ObservableBindingTrace
|
||||
import org.jetbrains.jet.lang.resolve.BindingTraceContext
|
||||
import org.jetbrains.jet.lang.resolve.calls.autocasts.DataFlowInfo
|
||||
import org.jetbrains.jet.lang.psi.JetParenthesizedExpression
|
||||
import com.intellij.codeInsight.template.TemplateBuilderImpl
|
||||
import com.intellij.codeInsight.template.impl.TemplateManagerImpl
|
||||
import org.jetbrains.jet.lang.resolve.DescriptorUtils
|
||||
import org.jetbrains.jet.lang.descriptors.ModuleDescriptor
|
||||
|
||||
import org.jetbrains.jet.analyzer.analyzeInContext
|
||||
import org.jetbrains.jet.lang.psi.JetCallExpression
|
||||
|
||||
public class AddForLoopIndicesIntention : JetSelfTargetingIntention<JetForExpression>(
|
||||
"add.for.loop.indices", javaClass()) {
|
||||
@@ -69,19 +64,20 @@ public class AddForLoopIndicesIntention : JetSelfTargetingIntention<JetForExpres
|
||||
if (element.getLoopParameter() == null) return false
|
||||
val range = element.getLoopRange() ?: return false
|
||||
if (range is JetDotQualifiedExpression) {
|
||||
val selector = (range as JetDotQualifiedExpression).getSelectorExpression() ?: return true
|
||||
if (selector.getText().equals("withIndices()")) return false
|
||||
val selector = range.getSelectorExpression() ?: return true
|
||||
if (selector.getText() == "withIndices()") return false
|
||||
}
|
||||
|
||||
val potentialExpression = JetPsiFactory.createExpression(element.getProject(), "${range.getText()}.withIndices()")
|
||||
val potentialExpression = JetPsiFactory.createExpression(element.getProject(),
|
||||
"${range.getText()}.withIndices()") as JetDotQualifiedExpression
|
||||
|
||||
val bindingContext = AnalyzerFacadeWithCache.getContextForElement(element)
|
||||
val scope = bindingContext[BindingContext.RESOLUTION_SCOPE, element] ?: return false
|
||||
val module = DescriptorUtils.getParentOfType(scope.getContainingDeclaration(), javaClass<ModuleDescriptor>())!!
|
||||
val components = InjectorForMacros(element.getProject(), module)
|
||||
val dataFlowInfo = bindingContext[BindingContext.NON_DEFAULT_EXPRESSION_DATA_FLOW, element] ?: DataFlowInfo.EMPTY
|
||||
val bindingTrace = ObservableBindingTrace(BindingTraceContext())
|
||||
val expressionType = components.getExpressionTypingServices()!!.getTypeInfo(scope, potentialExpression, TypeUtils.NO_EXPECTED_TYPE, dataFlowInfo, bindingTrace)
|
||||
return expressionType.getType() != null
|
||||
val functionSelector = potentialExpression.getSelectorExpression() as JetCallExpression
|
||||
val updatedContext = potentialExpression.analyzeInContext(scope)
|
||||
val callScope = updatedContext[BindingContext.RESOLVED_CALL, functionSelector.getCalleeExpression()!!] ?: return false
|
||||
val callFqName = DescriptorUtils.getFqNameSafe(callScope.getCandidateDescriptor())
|
||||
return callFqName.toString() == "kotlin.withIndices"
|
||||
}
|
||||
|
||||
}
|
||||
@@ -18,25 +18,16 @@ package org.jetbrains.jet.plugin.intentions
|
||||
|
||||
import org.jetbrains.jet.lang.psi.JetForExpression
|
||||
import com.intellij.openapi.editor.Editor
|
||||
import org.jetbrains.jet.lang.psi.JetPsiFactory
|
||||
import org.jetbrains.jet.lang.psi.JetDotQualifiedExpression
|
||||
import org.jetbrains.jet.lang.psi.JetPsiUtil
|
||||
import org.jetbrains.jet.lang.cfg.PseudocodeVariablesData.VariableUseState
|
||||
import org.jetbrains.jet.lang.cfg.pseudocode.PseudocodeUtil
|
||||
import org.jetbrains.jet.plugin.project.AnalyzerFacadeWithCache
|
||||
import org.jetbrains.jet.lang.psi.JetFile
|
||||
import org.jetbrains.jet.lang.resolve.ObservableBindingTrace
|
||||
import org.jetbrains.jet.lang.resolve.BindingTraceContext
|
||||
import org.jetbrains.jet.lang.cfg.JetFlowInformationProvider
|
||||
import org.jetbrains.jet.lang.cfg.JetControlFlowProcessor
|
||||
import com.intellij.debugger.jdi.LocalVariablesUtil
|
||||
import com.siyeh.ig.psiutils.VariableAccessUtils
|
||||
import com.google.dart.compiler.util.AstUtil
|
||||
import com.intellij.find.FindManager
|
||||
import com.intellij.find.impl.FindManagerImpl
|
||||
import com.intellij.usageView.UsageInfo
|
||||
import com.intellij.util.Processor
|
||||
import org.jetbrains.jet.plugin.findUsages.KotlinPropertyFindUsagesOptions
|
||||
import org.jetbrains.jet.lang.resolve.BindingContext
|
||||
import org.jetbrains.jet.lang.psi.JetCallExpression
|
||||
import org.jetbrains.jet.lang.resolve.DescriptorUtils
|
||||
|
||||
public class RemoveForLoopIndicesIntention : JetSelfTargetingIntention<JetForExpression>(
|
||||
"remove.for.loop.indices", javaClass()) {
|
||||
@@ -46,7 +37,8 @@ public class RemoveForLoopIndicesIntention : JetSelfTargetingIntention<JetForExp
|
||||
val parameters = parameter.getEntries()
|
||||
if (parameters.size() == 2) {
|
||||
parameter.replace(parameters[1])
|
||||
} else {
|
||||
}
|
||||
else {
|
||||
JetPsiUtil.deleteElementWithDelimiters(parameters[0])
|
||||
}
|
||||
|
||||
@@ -54,30 +46,25 @@ public class RemoveForLoopIndicesIntention : JetSelfTargetingIntention<JetForExp
|
||||
}
|
||||
|
||||
override fun isApplicableTo(element: JetForExpression): Boolean {
|
||||
if (element.getMultiParameter() == null) return false
|
||||
val multiParameter = element.getMultiParameter() ?: return false
|
||||
val range = element.getLoopRange() as? JetDotQualifiedExpression ?: return false
|
||||
val selector = range.getSelectorExpression() ?: return false
|
||||
val selector = range.getSelectorExpression() as? JetCallExpression ?: return false
|
||||
|
||||
if (!selector.textMatches("withIndices()")) return false
|
||||
|
||||
val indexVar = element.getMultiParameter()!!.getEntries()[0]
|
||||
val bindingContext = AnalyzerFacadeWithCache.getContextForElement(element)
|
||||
val callResolution = bindingContext[BindingContext.RESOLVED_CALL, selector.getCalleeExpression()!!] ?: return false
|
||||
val fqName = DescriptorUtils.getFqNameSafe(callResolution.getCandidateDescriptor())
|
||||
if (fqName.toString() != "kotlin.withIndices") return false
|
||||
|
||||
val indexVar = multiParameter.getEntries()[0]
|
||||
val findManager = FindManager.getInstance(element.getProject()) as FindManagerImpl
|
||||
val findHandler = findManager.getFindUsagesManager().getFindUsagesHandler(indexVar,false) ?: return false
|
||||
val findHandler = findManager.getFindUsagesManager().getFindUsagesHandler(indexVar, false) ?: return false
|
||||
val options = KotlinPropertyFindUsagesOptions(element.getProject())
|
||||
val usageCount = array(0)
|
||||
val usageFinderRunnable = object : Runnable {
|
||||
override fun run() {
|
||||
val processor = object : Processor<UsageInfo> {
|
||||
override fun process(t: UsageInfo?): Boolean {
|
||||
usageCount[0] = usageCount[0] + 1
|
||||
return true
|
||||
}
|
||||
}
|
||||
findHandler.processElementUsages(indexVar,processor,options)
|
||||
}
|
||||
}
|
||||
usageFinderRunnable.run()
|
||||
return usageCount[0] == 0
|
||||
var usageCount = 0
|
||||
val processorLambda: (UsageInfo?) -> Boolean = { t -> usageCount++; false }
|
||||
findHandler.processElementUsages(indexVar, processorLambda, options)
|
||||
return usageCount == 0
|
||||
}
|
||||
|
||||
}
|
||||
@@ -1,5 +1,5 @@
|
||||
//IS_APPLICABLE: FALSE
|
||||
//ERROR: Unresolved reference: withIndices
|
||||
// WITH_RUNTIME
|
||||
fun b(c: List<String>) {
|
||||
for ((<caret>indexVariable, d) in c.withIndices()) {
|
||||
|
||||
|
||||
@@ -0,0 +1,9 @@
|
||||
// WITH_RUNTIME
|
||||
// IS_APPLICABLE: FALSE
|
||||
fun String.withIndices(): Int = 42
|
||||
|
||||
fun foo(s: String) {
|
||||
for (<caret>a in s) {
|
||||
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,7 @@
|
||||
//IS_APPLICABLE: FALSE
|
||||
//ERROR: Unresolved reference: b
|
||||
fun foo() {
|
||||
for (a <caret>in b) {
|
||||
|
||||
}
|
||||
}
|
||||
+1
-1
@@ -1,5 +1,5 @@
|
||||
// IS_APPLICABLE: FALSE
|
||||
// ERROR: For-loop range must have an iterator() method
|
||||
// WITH_RUNTIME
|
||||
fun foo(bar: Map<String, String>) {
|
||||
for (a <caret>in bar) {
|
||||
|
||||
|
||||
@@ -0,0 +1,6 @@
|
||||
//WITH_RUNTIME
|
||||
fun foo(bar: IntArray) {
|
||||
for (<caret>a in bar) {
|
||||
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,6 @@
|
||||
//WITH_RUNTIME
|
||||
fun foo(bar: IntArray) {
|
||||
for ((index, a) in bar.withIndices()) {
|
||||
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,6 @@
|
||||
//WITH_RUNTIME
|
||||
fun foo(bar: Iterable<Int>) {
|
||||
for (<caret>a in bar) {
|
||||
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,6 @@
|
||||
//WITH_RUNTIME
|
||||
fun foo(bar: Iterable<Int>) {
|
||||
for ((index, a) in bar.withIndices()) {
|
||||
|
||||
}
|
||||
}
|
||||
@@ -1,4 +1,4 @@
|
||||
// ERROR: Unresolved reference: listOf
|
||||
// WITH_RUNTIME
|
||||
fun a() {
|
||||
val b = listOf(1,2,3,4,5)
|
||||
for (<caret>c : Int in b) {
|
||||
|
||||
@@ -1,4 +1,4 @@
|
||||
// ERROR: Unresolved reference: listOf
|
||||
// WITH_RUNTIME
|
||||
fun a() {
|
||||
val b = listOf(1,2,3,4,5)
|
||||
for ((index, c : Int) in b.withIndices()) {
|
||||
|
||||
@@ -0,0 +1,6 @@
|
||||
//WITH_RUNTIME
|
||||
fun foo(bar: Array<Object>) {
|
||||
for (<caret>a in bar) {
|
||||
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,6 @@
|
||||
//WITH_RUNTIME
|
||||
fun foo(bar: Array<Object>) {
|
||||
for ((index, a) in bar.withIndices()) {
|
||||
|
||||
}
|
||||
}
|
||||
@@ -1,4 +1,4 @@
|
||||
// ERROR: Unresolved reference: listOf
|
||||
// WITH_RUNTIME
|
||||
fun a() {
|
||||
val b = listOf(1,2,3,4,5)
|
||||
for (<caret>c in b) {
|
||||
|
||||
@@ -1,4 +1,4 @@
|
||||
// ERROR: Unresolved reference: listOf
|
||||
// WITH_RUNTIME
|
||||
fun a() {
|
||||
val b = listOf(1,2,3,4,5)
|
||||
for ((index, c) in b.withIndices()) {
|
||||
|
||||
@@ -1,6 +0,0 @@
|
||||
//ERROR: Unresolved reference: SortedMap
|
||||
fun a(b: SortedMap<Int, String>) {
|
||||
for (<caret>c in b) {
|
||||
|
||||
}
|
||||
}
|
||||
@@ -1,6 +0,0 @@
|
||||
//ERROR: Unresolved reference: SortedMap
|
||||
fun a(b: SortedMap<Int, String>) {
|
||||
for ((index, c) in b.withIndices()) {
|
||||
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,6 @@
|
||||
//WITH_RUNTIME
|
||||
fun foo(bar: Stream<String>) {
|
||||
for (<caret>a in bar) {
|
||||
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,6 @@
|
||||
//WITH_RUNTIME
|
||||
fun foo(bar: Stream<String>) {
|
||||
for ((index, a) in bar.withIndices()) {
|
||||
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,6 @@
|
||||
//WITH_RUNTIME
|
||||
fun foo(bar: String) {
|
||||
for (<caret>a in bar) {
|
||||
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,6 @@
|
||||
//WITH_RUNTIME
|
||||
fun foo(bar: String) {
|
||||
for ((index, a) in bar.withIndices()) {
|
||||
|
||||
}
|
||||
}
|
||||
@@ -1,7 +1,9 @@
|
||||
// IS_APPLICABLE: FALSE
|
||||
//ERROR: Unresolved reference: withIndices
|
||||
fun foo(b: List<Int>) {
|
||||
//WITH_RUNTIME
|
||||
|
||||
fun foo(b: List<Int>) : Int {
|
||||
for ((i, <caret>c) in b.withIndices()) {
|
||||
return i
|
||||
}
|
||||
return 0
|
||||
}
|
||||
+11
@@ -0,0 +1,11 @@
|
||||
// WITH_RUNTIME
|
||||
// IS_APPLICABLE: FALSE
|
||||
import java.util.LinkedList
|
||||
|
||||
fun Int.withIndices(): List<Pair<Int, Int>> = LinkedList<Pair<Int, Int>>()
|
||||
|
||||
fun foo(s: Int) {
|
||||
for ((index<caret>, a) in s.withIndices()) {
|
||||
|
||||
}
|
||||
}
|
||||
@@ -1,4 +1,4 @@
|
||||
//ERROR: Unresolved reference: withIndices
|
||||
//WITH_RUNTIME
|
||||
fun foo(bar: List<Int>) {
|
||||
for ((i : <caret>Int, b: Int) in bar.withIndices()) {
|
||||
|
||||
|
||||
@@ -1,4 +1,4 @@
|
||||
//ERROR: Unresolved reference: withIndices
|
||||
//WITH_RUNTIME
|
||||
fun foo(bar: List<Int>) {
|
||||
for (b: Int in bar) {
|
||||
|
||||
|
||||
@@ -1,4 +1,4 @@
|
||||
//ERROR: Unresolved reference: withIndices
|
||||
//WITH_RUNTIME
|
||||
fun foo(bar: List<String>) {
|
||||
for ((i,<caret>a) in bar.withIndices()) {
|
||||
|
||||
|
||||
+1
-1
@@ -1,4 +1,4 @@
|
||||
//ERROR: Unresolved reference: withIndices
|
||||
//WITH_RUNTIME
|
||||
fun foo(bar: List<String>) {
|
||||
for (a in bar) {
|
||||
|
||||
|
||||
+38
-3
@@ -4252,24 +4252,54 @@ public class CodeTransformationTestGenerated extends AbstractCodeTransformationT
|
||||
doTestAddForLoopIndicesIntention("idea/testData/intentions/addForLoopIndicesIntention/inapplicableExistingIndices.kt");
|
||||
}
|
||||
|
||||
@TestMetadata("inapplicableOverridenFunction.kt")
|
||||
public void testInapplicableOverridenFunction() throws Exception {
|
||||
doTestAddForLoopIndicesIntention("idea/testData/intentions/addForLoopIndicesIntention/inapplicableOverridenFunction.kt");
|
||||
}
|
||||
|
||||
@TestMetadata("inapplicableSyntaxError.kt")
|
||||
public void testInapplicableSyntaxError() throws Exception {
|
||||
doTestAddForLoopIndicesIntention("idea/testData/intentions/addForLoopIndicesIntention/inapplicableSyntaxError.kt");
|
||||
}
|
||||
|
||||
@TestMetadata("inapplicableUnorderedCollection.kt")
|
||||
public void testInapplicableUnorderedCollection() throws Exception {
|
||||
doTestAddForLoopIndicesIntention("idea/testData/intentions/addForLoopIndicesIntention/inapplicableUnorderedCollection.kt");
|
||||
}
|
||||
|
||||
@TestMetadata("intArray.kt")
|
||||
public void testIntArray() throws Exception {
|
||||
doTestAddForLoopIndicesIntention("idea/testData/intentions/addForLoopIndicesIntention/intArray.kt");
|
||||
}
|
||||
|
||||
@TestMetadata("iterable.kt")
|
||||
public void testIterable() throws Exception {
|
||||
doTestAddForLoopIndicesIntention("idea/testData/intentions/addForLoopIndicesIntention/iterable.kt");
|
||||
}
|
||||
|
||||
@TestMetadata("listWithType.kt")
|
||||
public void testListWithType() throws Exception {
|
||||
doTestAddForLoopIndicesIntention("idea/testData/intentions/addForLoopIndicesIntention/listWithType.kt");
|
||||
}
|
||||
|
||||
@TestMetadata("objectArray.kt")
|
||||
public void testObjectArray() throws Exception {
|
||||
doTestAddForLoopIndicesIntention("idea/testData/intentions/addForLoopIndicesIntention/objectArray.kt");
|
||||
}
|
||||
|
||||
@TestMetadata("simpleIntList.kt")
|
||||
public void testSimpleIntList() throws Exception {
|
||||
doTestAddForLoopIndicesIntention("idea/testData/intentions/addForLoopIndicesIntention/simpleIntList.kt");
|
||||
}
|
||||
|
||||
@TestMetadata("simpleSortedMap.kt")
|
||||
public void testSimpleSortedMap() throws Exception {
|
||||
doTestAddForLoopIndicesIntention("idea/testData/intentions/addForLoopIndicesIntention/simpleSortedMap.kt");
|
||||
@TestMetadata("stream.kt")
|
||||
public void testStream() throws Exception {
|
||||
doTestAddForLoopIndicesIntention("idea/testData/intentions/addForLoopIndicesIntention/stream.kt");
|
||||
}
|
||||
|
||||
@TestMetadata("string.kt")
|
||||
public void testString() throws Exception {
|
||||
doTestAddForLoopIndicesIntention("idea/testData/intentions/addForLoopIndicesIntention/string.kt");
|
||||
}
|
||||
|
||||
}
|
||||
@@ -4290,6 +4320,11 @@ public class CodeTransformationTestGenerated extends AbstractCodeTransformationT
|
||||
doTestRemoveForLoopIndicesIntention("idea/testData/intentions/removeForLoopIndicesIntention/inapplicableIndexUse.kt");
|
||||
}
|
||||
|
||||
@TestMetadata("inapplicableOverridenFunction.kt")
|
||||
public void testInapplicableOverridenFunction() throws Exception {
|
||||
doTestRemoveForLoopIndicesIntention("idea/testData/intentions/removeForLoopIndicesIntention/inapplicableOverridenFunction.kt");
|
||||
}
|
||||
|
||||
@TestMetadata("loopWithType.kt")
|
||||
public void testLoopWithType() throws Exception {
|
||||
doTestRemoveForLoopIndicesIntention("idea/testData/intentions/removeForLoopIndicesIntention/loopWithType.kt");
|
||||
|
||||
Reference in New Issue
Block a user