Skip to content

Commit b27f35f

Browse files
committed
Added rename
1 parent b7f1b66 commit b27f35f

File tree

4 files changed

+63
-2
lines changed

4 files changed

+63
-2
lines changed

server/src/main/kotlin/org/javacs/kt/KotlinLanguageServer.kt

+1
Original file line numberDiff line numberDiff line change
@@ -76,6 +76,7 @@ class KotlinLanguageServer : LanguageServer, LanguageClientAware, Closeable {
7676
serverCapabilities.workspace.workspaceFolders.supported = true
7777
serverCapabilities.workspace.workspaceFolders.changeNotifications = Either.forRight(true)
7878
serverCapabilities.hoverProvider = Either.forLeft(true)
79+
serverCapabilities.renameProvider = Either.forLeft(true)
7980
serverCapabilities.completionProvider = CompletionOptions(false, listOf("."))
8081
serverCapabilities.signatureHelpProvider = SignatureHelpOptions(listOf("(", ","))
8182
serverCapabilities.definitionProvider = Either.forLeft(true)

server/src/main/kotlin/org/javacs/kt/KotlinTextDocumentService.kt

+4-2
Original file line numberDiff line numberDiff line change
@@ -26,6 +26,7 @@ import org.javacs.kt.util.parseURI
2626
import org.javacs.kt.util.describeURI
2727
import org.javacs.kt.util.describeURIs
2828
import org.javacs.kt.commands.JAVA_TO_KOTLIN_COMMAND
29+
import org.javacs.kt.rename.renameSymbol
2930
import org.jetbrains.kotlin.resolve.diagnostics.Diagnostics
3031
import java.net.URI
3132
import java.io.Closeable
@@ -144,8 +145,9 @@ class KotlinTextDocumentService(
144145
TODO("not implemented")
145146
}
146147

147-
override fun rename(params: RenameParams): CompletableFuture<WorkspaceEdit> {
148-
TODO("not implemented")
148+
override fun rename(params: RenameParams) = async.compute {
149+
val (file, cursor) = recover(params, Recompile.NEVER)
150+
renameSymbol(file, cursor, sp, params.newName)
149151
}
150152

151153
override fun completion(position: CompletionParams) = async.compute {

server/src/main/kotlin/org/javacs/kt/references/FindReferences.kt

+14
Original file line numberDiff line numberDiff line change
@@ -4,6 +4,7 @@ import org.eclipse.lsp4j.Location
44
import org.javacs.kt.LOG
55
import org.javacs.kt.SourcePath
66
import org.javacs.kt.position.location
7+
import org.javacs.kt.position.toURIString
78
import org.javacs.kt.util.emptyResult
89
import org.javacs.kt.util.findParent
910
import org.javacs.kt.util.preOrderTraversal
@@ -31,9 +32,22 @@ fun findReferences(file: Path, cursor: Int, sp: SourcePath): List<Location> {
3132
.sortedWith(compareBy({ it.getUri() }, { it.getRange().getStart().getLine() }))
3233
}
3334

35+
fun findReferences(declaration: KtNamedDeclaration, sp: SourcePath): List<Location> {
36+
return doFindReferences(declaration, sp)
37+
.map { location(it) }
38+
.filterNotNull()
39+
.toList()
40+
.sortedWith(compareBy({ it.getUri() }, { it.getRange().getStart().getLine() }))
41+
}
42+
3443
private fun doFindReferences(file: Path, cursor: Int, sp: SourcePath): Collection<KtElement> {
3544
val recover = sp.currentVersion(file.toUri())
3645
val element = recover.elementAtPoint(cursor)?.findParent<KtNamedDeclaration>() ?: return emptyResult("No declaration at ${recover.describePosition(cursor)}")
46+
return doFindReferences(element, sp)
47+
}
48+
49+
private fun doFindReferences(element: KtNamedDeclaration, sp: SourcePath): Collection<KtElement> {
50+
val recover = sp.currentVersion(element.containingFile.toPath().toUri())
3751
val declaration = recover.compile[BindingContext.DECLARATION_TO_DESCRIPTOR, element] ?: return emptyResult("Declaration ${element.fqName} has no descriptor")
3852
val maybes = possibleReferences(declaration, sp).map { it.toPath() }
3953
LOG.debug("Scanning {} files for references to {}", maybes.size, element.fqName)
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,44 @@
1+
package org.javacs.kt.rename
2+
3+
import org.eclipse.lsp4j.*
4+
import org.eclipse.lsp4j.jsonrpc.messages.Either
5+
import org.javacs.kt.CompiledFile
6+
import org.javacs.kt.SourcePath
7+
import org.javacs.kt.position.location
8+
import org.javacs.kt.references.findReferences
9+
import org.jetbrains.kotlin.js.resolve.diagnostics.findPsi
10+
import org.jetbrains.kotlin.psi.KtNamedDeclaration
11+
12+
fun renameSymbol(file: CompiledFile, cursor: Int, sp: SourcePath, newName: String): WorkspaceEdit? {
13+
val (declaration, location) = findDeclaration(file, cursor) ?: return null
14+
return declaration.let {
15+
val declarationEdit = Either.forLeft<TextDocumentEdit, ResourceOperation>(TextDocumentEdit(
16+
VersionedTextDocumentIdentifier().apply { uri = location.uri },
17+
listOf(TextEdit(location.range, newName))
18+
))
19+
20+
val referenceEdits = findReferences(declaration, sp).map {
21+
Either.forLeft<TextDocumentEdit, ResourceOperation>(TextDocumentEdit(
22+
VersionedTextDocumentIdentifier().apply { uri = it.uri },
23+
listOf(TextEdit(it.range, newName))
24+
))
25+
}
26+
27+
WorkspaceEdit(listOf(declarationEdit) + referenceEdits)
28+
}
29+
}
30+
31+
private fun findDeclaration(file: CompiledFile, cursor: Int): Pair<KtNamedDeclaration, Location>? {
32+
val (_, target) = file.referenceAtPoint(cursor) ?: return null
33+
val psi = target.findPsi()
34+
35+
return if (psi is KtNamedDeclaration) {
36+
psi.nameIdentifier?.let {
37+
location(it)?.let { location ->
38+
Pair(psi, location)
39+
}
40+
}
41+
} else {
42+
null
43+
}
44+
}

0 commit comments

Comments
 (0)