Skip to content

Commit 3c40306

Browse files
committed
feat: console command runners
1 parent 546ce1c commit 3c40306

11 files changed

+275
-1
lines changed
Lines changed: 12 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,12 @@
1+
package com.github.tempest.framework.console.run
2+
3+
import com.intellij.execution.lineMarker.RunLineMarkerContributor
4+
import com.intellij.psi.PsiElement
5+
import com.jetbrains.php.lang.psi.elements.Method
6+
7+
class ConsoleCommandLineMarkerProvider : RunLineMarkerContributor() {
8+
override fun getInfo(element: PsiElement) = when {
9+
element !is Method -> null
10+
else -> Info(TempestRunTargetAction(element))
11+
}
12+
}
Lines changed: 35 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,35 @@
1+
package com.github.tempest.framework.console.run
2+
3+
import com.intellij.execution.configurations.ConfigurationFactory
4+
import com.intellij.execution.configurations.RunConfigurationOptions
5+
import com.intellij.openapi.project.Project
6+
import com.jetbrains.php.config.commandLine.PhpCommandSettings
7+
import com.jetbrains.php.run.PhpCommandLineRunConfiguration
8+
9+
class TempestConsoleCommandRunConfiguration(
10+
project: Project,
11+
factory: ConfigurationFactory,
12+
name: String
13+
) : PhpCommandLineRunConfiguration<TempestConsoleCommandRunConfigurationSettings>(project, factory, name) {
14+
override fun fillCommandSettings(
15+
envs: Map<String, String>,
16+
command: PhpCommandSettings
17+
) {
18+
val commandName = settings.commandName ?: return
19+
command.setScript("tempest", false)
20+
command.addArgument(commandName)
21+
22+
command.importCommandLineSettings(settings.commandLineSettings, command.workingDirectory)
23+
command.addEnvs(envs)
24+
}
25+
26+
override fun getOptions() = super.getOptions() as TempestConsoleCommandRunConfigurationSettings
27+
override fun getOptionsClass(): Class<out RunConfigurationOptions> {
28+
return TempestConsoleCommandRunConfigurationSettings::class.java
29+
}
30+
31+
override fun getConfigurationEditor() = TempestConsoleCommandSettingsEditor(project)
32+
33+
override fun createSettings() = TempestConsoleCommandRunConfigurationSettings().apply {
34+
}
35+
}
Lines changed: 37 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,37 @@
1+
package com.github.tempest.framework.console.run
2+
3+
import com.intellij.execution.configurations.LocatableRunConfigurationOptions
4+
import com.jetbrains.php.run.PhpCommandLineSettings
5+
import com.jetbrains.php.run.PhpRunConfigurationSettings
6+
7+
class TempestConsoleCommandRunConfigurationSettings : PhpRunConfigurationSettings, LocatableRunConfigurationOptions() {
8+
private val myCommandName = string("").provideDelegate(this, "commandName")
9+
private val myBinary = string("./tempest").provideDelegate(this, "binary")
10+
private val myWorkingDirectory = string("").provideDelegate(this, "binary")
11+
12+
var commandName: String?
13+
get() = myCommandName.getValue(this)
14+
set(scriptName) {
15+
myCommandName.setValue(this, scriptName)
16+
}
17+
18+
var binary: String?
19+
get() = myBinary.getValue(this)
20+
set(scriptName) {
21+
myBinary.setValue(this, scriptName)
22+
}
23+
24+
var documentRoot: String?
25+
get() = myBinary.getValue(this)
26+
set(scriptName) {
27+
myBinary.setValue(this, scriptName)
28+
}
29+
30+
var commandLineSettings = PhpCommandLineSettings()
31+
32+
override fun getWorkingDirectory() = myWorkingDirectory.getValue(this)
33+
34+
override fun setWorkingDirectory(p0: String?) {
35+
myWorkingDirectory.setValue(this, p0)
36+
}
37+
}
Lines changed: 30 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,30 @@
1+
package com.github.tempest.framework.console.run
2+
3+
import com.github.tempest.framework.TempestIcons
4+
import com.intellij.execution.configurations.ConfigurationFactory
5+
import com.intellij.execution.configurations.ConfigurationTypeBase
6+
import com.intellij.openapi.project.Project
7+
8+
class TempestConsoleCommandRunConfigurationType : ConfigurationTypeBase(
9+
ID,
10+
"Tempest Command",
11+
"Runs console command",
12+
TempestIcons.TEMPEST,
13+
) {
14+
init {
15+
addFactory(object : ConfigurationFactory(this) {
16+
override fun getId() = ID
17+
18+
override fun createTemplateConfiguration(project: Project) =
19+
TempestConsoleCommandRunConfiguration(project, this, "Tempest")
20+
21+
override fun getOptionsClass() = TempestConsoleCommandRunConfigurationSettings::class.java
22+
})
23+
}
24+
25+
companion object {
26+
const val ID = "TempestConsoleCommandRunConfiguration"
27+
28+
val INSTANCE = TempestConsoleCommandRunConfigurationType()
29+
}
30+
}
Lines changed: 58 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,58 @@
1+
package com.github.tempest.framework.console.run
2+
3+
import com.intellij.openapi.options.SettingsEditor
4+
import com.intellij.openapi.ui.DialogPanel
5+
import com.intellij.ui.dsl.builder.Align
6+
import com.intellij.ui.dsl.builder.LabelPosition
7+
import com.intellij.ui.dsl.builder.TopGap
8+
import com.intellij.ui.dsl.builder.panel
9+
import com.jetbrains.php.run.PhpCommandLineConfigurationEditor
10+
import javax.swing.JPanel
11+
import javax.swing.JTextField
12+
13+
private fun PhpCommandLineConfigurationEditor.getMainPanel(): JPanel? {
14+
val reflection = PhpCommandLineConfigurationEditor::class.java.getDeclaredField("myMainPanel")
15+
reflection.isAccessible = true
16+
return reflection.get(this) as JPanel?
17+
}
18+
19+
class TempestConsoleCommandSettingsEditor private constructor(): SettingsEditor<TempestConsoleCommandRunConfiguration>() {
20+
private val commandNameField = JTextField()
21+
private val phpCommandLineConfigurationEditor = PhpCommandLineConfigurationEditor()
22+
23+
private lateinit var myPanel: DialogPanel
24+
25+
constructor(project: com.intellij.openapi.project.Project) : this() {
26+
myPanel = panel {
27+
row {
28+
cell(commandNameField)
29+
.label("Command:", LabelPosition.LEFT)
30+
.align(Align.FILL)
31+
}.topGap(TopGap.MEDIUM)
32+
33+
row {
34+
phpCommandLineConfigurationEditor.init(project, true)
35+
phpCommandLineConfigurationEditor.getMainPanel()?.apply {
36+
scrollCell(this).align(Align.FILL)
37+
}
38+
}.topGap(TopGap.MEDIUM)
39+
}
40+
}
41+
42+
override fun resetEditorFrom(tempestConsoleCommandRunConfiguration: TempestConsoleCommandRunConfiguration) {
43+
val settings = tempestConsoleCommandRunConfiguration.settings
44+
45+
myPanel.reset()
46+
commandNameField.text = settings.commandName
47+
phpCommandLineConfigurationEditor.resetEditorFrom(settings.commandLineSettings)
48+
}
49+
50+
override fun applyEditorTo(tempestConsoleCommandRunConfiguration: TempestConsoleCommandRunConfiguration) {
51+
val settings = tempestConsoleCommandRunConfiguration.settings
52+
53+
settings.commandName = commandNameField.text
54+
phpCommandLineConfigurationEditor.applyEditorTo(settings.commandLineSettings)
55+
}
56+
57+
override fun createEditor() = myPanel
58+
}
Lines changed: 13 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,13 @@
1+
package com.github.tempest.framework.console.run
2+
3+
import com.intellij.execution.configurations.ConfigurationFactory
4+
import com.intellij.openapi.project.Project
5+
6+
class TempestRunConfigurationFactory(private val runConfigurationType: TempestConsoleCommandRunConfigurationType) :
7+
ConfigurationFactory(runConfigurationType) {
8+
override fun getId() = TempestConsoleCommandRunConfigurationType.ID
9+
override fun getName() = runConfigurationType.displayName
10+
11+
override fun createTemplateConfiguration(project: Project) =
12+
TempestConsoleCommandRunConfiguration(project, this, "name")
13+
}
Lines changed: 35 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,35 @@
1+
package com.github.tempest.framework.console.run
2+
3+
import com.github.tempest.framework.php.getConsoleCommandName
4+
import com.intellij.execution.actions.ConfigurationContext
5+
import com.intellij.execution.actions.LazyRunConfigurationProducer
6+
import com.intellij.openapi.util.Ref
7+
import com.intellij.psi.PsiElement
8+
import com.jetbrains.php.lang.psi.elements.Method
9+
10+
class TempestRunConfigurationProducer : LazyRunConfigurationProducer<TempestConsoleCommandRunConfiguration>() {
11+
override fun setupConfigurationFromContext(
12+
configuration: TempestConsoleCommandRunConfiguration,
13+
context: ConfigurationContext,
14+
sourceElement: Ref<PsiElement>
15+
): Boolean {
16+
val element = context.psiLocation as? Method ?: return false
17+
18+
configuration.settings.commandName = element.getConsoleCommandName() ?: return false
19+
configuration.name = "tempest ${configuration.settings.commandName}"
20+
21+
return true
22+
}
23+
24+
override fun isConfigurationFromContext(
25+
configuration: TempestConsoleCommandRunConfiguration,
26+
context: ConfigurationContext
27+
): Boolean {
28+
val method = context.psiLocation as? Method ?: return false
29+
30+
return configuration.settings.commandName == method.getConsoleCommandName()
31+
}
32+
33+
override fun getConfigurationFactory() =
34+
TempestRunConfigurationFactory(TempestConsoleCommandRunConfigurationType.INSTANCE)
35+
}
Lines changed: 34 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,34 @@
1+
package com.github.tempest.framework.console.run
2+
3+
import com.github.tempest.framework.TempestBundle
4+
import com.intellij.execution.Executor
5+
import com.intellij.execution.Location
6+
import com.intellij.execution.PsiLocation
7+
import com.intellij.execution.RunManagerEx
8+
import com.intellij.execution.actions.ConfigurationContext
9+
import com.intellij.execution.runners.ExecutionUtil
10+
import com.intellij.icons.AllIcons
11+
import com.intellij.openapi.actionSystem.AnAction
12+
import com.intellij.openapi.actionSystem.AnActionEvent
13+
import com.intellij.openapi.actionSystem.impl.SimpleDataContext
14+
import com.jetbrains.php.lang.psi.elements.Method
15+
16+
internal class TempestRunTargetAction(private val target: Method) : AnAction() {
17+
init {
18+
templatePresentation.setText(TempestBundle.message("action.run.target.text", target.name), false)
19+
templatePresentation.description = TempestBundle.message("action.run.target.description", target.name)
20+
templatePresentation.icon = AllIcons.Actions.Execute
21+
}
22+
23+
override fun actionPerformed(event: AnActionEvent) {
24+
val dataContext = SimpleDataContext.getSimpleContext(Location.DATA_KEY, PsiLocation(target), event.dataContext)
25+
26+
val context = ConfigurationContext.getFromContext(dataContext, event.place)
27+
28+
val producer = TempestRunConfigurationProducer()
29+
val configuration = producer.findOrCreateConfigurationFromContext(context)?.configurationSettings ?: return
30+
31+
(context.runManager as RunManagerEx).setTemporaryConfiguration(configuration)
32+
ExecutionUtil.runConfiguration(configuration, Executor.EXECUTOR_EXTENSION_NAME.extensionList.first())
33+
}
34+
}

src/main/kotlin/com/github/tempest/framework/php/mixin.kt

Lines changed: 14 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,10 @@
11
package com.github.tempest.framework.php
22

3+
import com.github.tempest.framework.TempestFrameworkClasses
4+
import com.intellij.openapi.util.text.StringUtil
35
import com.intellij.psi.util.PsiTreeUtil
46
import com.jetbrains.php.lang.psi.PhpFile
7+
import com.jetbrains.php.lang.psi.elements.Method
58
import com.jetbrains.php.lang.psi.elements.Variable
69

710
fun PhpFile.getPhpViewVariables(): Set<Variable> {
@@ -11,3 +14,14 @@ fun PhpFile.getPhpViewVariables(): Set<Variable> {
1114
.distinctBy { it.name }
1215
.toSet()
1316
}
17+
18+
fun Method.getConsoleCommandName(): String? {
19+
return this
20+
.getAttributes(TempestFrameworkClasses.ConsoleCommand)
21+
.firstOrNull()
22+
?.arguments
23+
?.run { this.find { it.name == "name" } ?: firstOrNull() }
24+
?.argument
25+
?.value
26+
?.run { StringUtil.unquoteString(this) }
27+
}

src/main/resources/META-INF/plugin.xml

Lines changed: 5 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -10,6 +10,11 @@
1010
<resource-bundle>messages.TempestBundle</resource-bundle>
1111

1212
<extensions defaultExtensionNs="com.intellij">
13+
<configurationType
14+
implementation="com.github.tempest.framework.console.run.TempestConsoleCommandRunConfigurationType"/>
15+
<runLineMarkerContributor
16+
language="PHP"
17+
implementationClass="com.github.tempest.framework.console.run.ConsoleCommandLineMarkerProvider" />
1318
<multiHostInjector
1419
implementation="com.github.tempest.framework.views.injection.PHPLanguageInjector"/>
1520
<webSymbols.webTypes
@@ -35,6 +40,5 @@
3540
implementation="com.github.tempest.framework.router.index.RouterActionsIndex" />
3641
</extensions>
3742
<extensions defaultExtensionNs="com.jetbrains.php">
38-
3943
</extensions>
4044
</idea-plugin>

0 commit comments

Comments
 (0)