Skip to content

Commit

Permalink
Formatting based on external formatter
Browse files Browse the repository at this point in the history
  • Loading branch information
cottand committed Jun 1, 2024
1 parent b25d38d commit e2d42bc
Show file tree
Hide file tree
Showing 6 changed files with 337 additions and 25 deletions.
107 changes: 107 additions & 0 deletions src/main/java/org/nixos/idea/format/NixExternalFormatter.java
Original file line number Diff line number Diff line change
@@ -0,0 +1,107 @@
package org.nixos.idea.format;

import com.intellij.execution.ExecutionException;
import com.intellij.execution.configurations.GeneralCommandLine;
import com.intellij.execution.process.CapturingProcessAdapter;
import com.intellij.execution.process.OSProcessHandler;
import com.intellij.execution.process.ProcessEvent;
import com.intellij.formatting.service.AsyncDocumentFormattingService;
import com.intellij.formatting.service.AsyncFormattingRequest;
import com.intellij.openapi.util.NlsSafe;
import com.intellij.psi.PsiFile;
import com.intellij.util.execution.ParametersListUtil;
import org.jetbrains.annotations.NonNls;
import org.jetbrains.annotations.NotNull;
import org.jetbrains.annotations.Nullable;
import org.nixos.idea.file.NixFile;
import org.nixos.idea.settings.NixLangSettings;

import java.io.*;
import java.nio.charset.StandardCharsets;
import java.nio.file.Files;
import java.util.EnumSet;
import java.util.List;
import java.util.Scanner;
import java.util.Set;

public final class NixExternalFormatter extends AsyncDocumentFormattingService {

@Override
protected @NotNull String getNotificationGroupId() {
return "NixIDEA";
}

@Override
protected @NotNull @NlsSafe String getName() {
return "NixIDEA";
}

@Override
public @NotNull Set<Feature> getFeatures() {
return EnumSet.noneOf(Feature.class);
}

@Override
public boolean canFormat(@NotNull PsiFile psiFile) {
return psiFile instanceof NixFile;
}


@Override
protected @Nullable FormattingTask createFormattingTask(@NotNull AsyncFormattingRequest request) {
NixLangSettings nixSettings = NixLangSettings.getInstance();
System.out.println("started fmt task");
if (!nixSettings.isFormatEnabled()) {
return null;
}

var ioFile = request.getIOFile();
if (ioFile == null) return null;

@NonNls
var command = nixSettings.getFormatCommand();
List<String> argv = ParametersListUtil.parse(command, false, true);

try {
var commandLine = new GeneralCommandLine(argv);

OSProcessHandler handler = new OSProcessHandler(commandLine.withCharset(StandardCharsets.UTF_8));
OutputStream processInput = handler.getProcessInput();
Files.copy(ioFile.toPath(), processInput);
processInput.flush();
processInput.close();
return new FormattingTask() {
@Override
public void run() {
handler.addProcessListener(new CapturingProcessAdapter() {

@Override
public void processTerminated(@NotNull ProcessEvent event) {
int exitCode = event.getExitCode();
if (exitCode == 0) {
request.onTextReady(getOutput().getStdout());
} else {
request.onError("NixIDEA", getOutput().getStderr());
}
}
});
handler.startNotify();
}

@Override
public boolean cancel() {
handler.destroyProcess();
return true;
}

@Override
public boolean isRunUnderProgress() {
return true;
}
};
} catch (ExecutionException | IOException e) {
request.onError("NixIDEA", e.getMessage());
return null;
}
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -18,8 +18,15 @@

import javax.swing.JComponent;
import javax.swing.JPanel;
import java.util.List;

public class NixLspSettingsConfigurable implements SearchableConfigurable, Configurable.Beta {
private static final List<CommandSuggestionsPopup.Suggestion> BUILTIN_SUGGESTIONS = List.of(
CommandSuggestionsPopup.Suggestion.builtin("<html>Use <b>nil</b> from nixpkgs</html>",
"nix --extra-experimental-features \"nix-command flakes\" run nixpkgs#nil"),
CommandSuggestionsPopup.Suggestion.builtin("<html>Use <b>nixd</b> from nixpkgs</html>",
"nix --extra-experimental-features \"nix-command flakes\" run nixpkgs#nixd")
);

private @Nullable JBCheckBox myEnabled;
private @Nullable RawCommandLineEditor myCommand;
Expand All @@ -43,7 +50,7 @@ public class NixLspSettingsConfigurable implements SearchableConfigurable, Confi
myCommand.getEditorField().getEmptyText().setText("Command to start Language Server");
myCommand.getEditorField().getAccessibleContext().setAccessibleName("Command to start Language Server");
myCommand.getEditorField().setMargin(myEnabled.getMargin());
new CommandSuggestionsPopup(myCommand, NixLspSettings.getInstance().getCommandHistory()).install();
new CommandSuggestionsPopup(myCommand, NixLspSettings.getInstance().getCommandHistory(), BUILTIN_SUGGESTIONS).install();

return FormBuilder.createFormBuilder()
.addComponent(myEnabled)
Expand Down
32 changes: 16 additions & 16 deletions src/main/java/org/nixos/idea/lsp/ui/CommandSuggestionsPopup.java
Original file line number Diff line number Diff line change
Expand Up @@ -37,20 +37,19 @@
import java.util.stream.Stream;

public final class CommandSuggestionsPopup {
// Implementation partially inspired by TextCompletionField

private static final List<Suggestion> BUILDIN_SUGGESTIONS = List.of(
Suggestion.builtin("<html>Use <b>nil</b> from nixpkgs</html>",
"nix --extra-experimental-features \"nix-command flakes\" run nixpkgs#nil"),
Suggestion.builtin("<html>Use <b>nixd</b> from nixpkgs</html>",
"nix --extra-experimental-features \"nix-command flakes\" run nixpkgs#nixd")
);
// Implementation partially inspired by TextCompletionField

private final @NotNull ExpandableTextField myEditor;
private final @NotNull Collection<String> myHistory;
private @Nullable ListPopup myPopup;
private final @NotNull List<Suggestion> mySuggestions;

public CommandSuggestionsPopup(@NotNull RawCommandLineEditor commandLineEditor, @NotNull Collection<String> history) {
public CommandSuggestionsPopup(@NotNull RawCommandLineEditor commandLineEditor,
@NotNull Collection<String> history,
@NotNull List<Suggestion> suggestions
) {
mySuggestions = suggestions;
myEditor = commandLineEditor.getEditorField();
myHistory = history;
}
Expand All @@ -72,7 +71,7 @@ public void install() {

public void show() {
if (myPopup == null) {
myPopup = new MyListPopup();
myPopup = new MyListPopup(mySuggestions);
myPopup.showUnderneathOf(myEditor);
}
}
Expand Down Expand Up @@ -110,8 +109,8 @@ public void caretUpdate(CaretEvent e) {
}

private final class MyListPopup extends ListPopupImpl implements JBPopupListener {
private MyListPopup() {
super(null, new MyListPopupStep());
private MyListPopup(List<Suggestion> suggestions) {
super(null, new MyListPopupStep(suggestions));
// Disable focus in popup, so that the text field stays in focus.
setRequestFocus(false);
// Prevent the popup from overriding the paste-action.
Expand All @@ -127,7 +126,8 @@ protected void process(KeyEvent aEvent) {
switch (aEvent.getKeyCode()) {
// Do no handle left and right key,
// as it would prevent their usage in the text field while the popup is open.
case KeyEvent.VK_LEFT, KeyEvent.VK_RIGHT -> {}
case KeyEvent.VK_LEFT, KeyEvent.VK_RIGHT -> {
}
default -> super.process(aEvent);
}
}
Expand All @@ -138,13 +138,13 @@ public void onClosed(@NotNull LightweightWindowEvent event) {
}
}

private record Suggestion(
public record Suggestion(
@NotNull Icon icon,
@NotNull String primaryText,
@Nullable String secondaryText,
@NotNull String command
) {
static @NotNull Suggestion builtin(@NotNull String name, @NotNull String command) {
public static @NotNull Suggestion builtin(@NotNull String name, @NotNull String command) {
return new Suggestion(AllIcons.Actions.Lightning, name, command, command);
}

Expand All @@ -161,9 +161,9 @@ public String toString() {

private final class MyListPopupStep extends BaseListPopupStep<Suggestion> implements ListPopupStepEx<Suggestion> {

public MyListPopupStep() {
public MyListPopupStep(List<Suggestion> suggestions) {
super(null, Stream.concat(
BUILDIN_SUGGESTIONS.stream(),
suggestions.stream(),
myHistory.stream().map(Suggestion::history)
).toList());
}
Expand Down
78 changes: 78 additions & 0 deletions src/main/java/org/nixos/idea/settings/NixLangSettings.java
Original file line number Diff line number Diff line change
@@ -0,0 +1,78 @@
package org.nixos.idea.settings;

import com.intellij.openapi.application.ApplicationManager;
import com.intellij.openapi.components.PersistentStateComponent;
import com.intellij.openapi.components.RoamingType;
import com.intellij.openapi.components.State;
import com.intellij.openapi.components.Storage;
import org.jetbrains.annotations.NotNull;

import java.util.ArrayDeque;
import java.util.Collection;
import java.util.Collections;
import java.util.Deque;

@State(name = "NixLangSettings", storages = @Storage(value = "nix-idea-tools.xml", roamingType = RoamingType.DISABLED))
public final class NixLangSettings implements PersistentStateComponent<NixLangSettings.State> {

// TODO: Use RoamingType.LOCAL with 2024.1

// Documentation:
// https://plugins.jetbrains.com/docs/intellij/persisting-state-of-components.html

private static final int MAX_HISTORY_SIZE = 5;

private @NotNull State myState = new State();

public static @NotNull NixLangSettings getInstance() {
return ApplicationManager.getApplication().getService(NixLangSettings.class);
}

public boolean isFormatEnabled() {
return myState.formatEnabled;
}

public void setFormatEnabled(boolean enabled) {
myState.formatEnabled = enabled;
}

public @NotNull String getFormatCommand() {
return myState.formatCommand;
}

public void setFormatCommand(@NotNull String command) {
myState.formatCommand = command;
addFormatCommandToHistory(command);
}

public @NotNull Collection<String> getCommandHistory() {
return Collections.unmodifiableCollection(myState.formatCommandHistory);
}

private void addFormatCommandToHistory(@NotNull String command) {
Deque<String> history = myState.formatCommandHistory;
history.remove(command);
history.addFirst(command);
while (history.size() > MAX_HISTORY_SIZE) {
history.removeLast();
}
}

@SuppressWarnings("ClassEscapesDefinedScope")
@Override
public void loadState(@NotNull State state) {
myState = state;
}

@SuppressWarnings("ClassEscapesDefinedScope")
@Override
public @NotNull State getState() {
return myState;
}

static final class State {
public boolean formatEnabled = false;
public @NotNull String formatCommand = "";
public Deque<String> formatCommandHistory = new ArrayDeque<>();
}
}
Loading

0 comments on commit e2d42bc

Please sign in to comment.