All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.gradle.gradleplugin.userinterface.swing.generic.OutputPanel Maven / Gradle / Ivy

There is a newer version: 8.11.1
Show newest version
/*
 * Copyright 2010 the original author or authors.
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *      http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */
package org.gradle.gradleplugin.userinterface.swing.generic;

import org.gradle.BuildResult;
import org.gradle.foundation.ipc.gradle.ExecuteGradleCommandServerProtocol;
import org.gradle.foundation.output.FileLink;
import org.gradle.foundation.output.FileLinkDefinitionLord;
import org.gradle.gradleplugin.foundation.GradlePluginLord;
import org.gradle.gradleplugin.foundation.favorites.FavoriteTask;
import org.gradle.gradleplugin.foundation.request.RefreshTaskListRequest;
import org.gradle.gradleplugin.foundation.request.Request;
import org.gradle.gradleplugin.userinterface.AlternateUIInteraction;
import org.gradle.gradleplugin.userinterface.swing.common.SearchPanel;
import org.gradle.gradleplugin.userinterface.swing.common.TextPaneSearchInteraction;
import org.gradle.api.logging.configuration.ShowStacktrace;

import javax.swing.*;
import javax.swing.text.AttributeSet;
import javax.swing.text.StyleConstants;
import javax.swing.text.StyleContext;
import java.awt.*;
import java.awt.event.ActionEvent;
import java.awt.event.MouseAdapter;
import java.awt.event.MouseEvent;
import java.io.File;
import java.text.SimpleDateFormat;
import java.util.Calendar;

/**
 * This is a panel that displays the results of executing a gradle command. It shows gradle's output as well as progress.
 */
public class OutputPanel extends JPanel implements ExecuteGradleCommandServerProtocol.ExecutionInteraction {

    private GradlePluginLord gradlePluginLord;
    private OutputPanelParent parent;
    private AlternateUIInteraction alternateUIInteraction;

    private JPanel gradleOutputTextPanel;
    private OutputTextPane gradleOutputTextPane;

    private JPanel progressPanel;
    private JLabel progressLabel;
    private JProgressBar progressBar;

    private JPanel statusPanel;
    private JLabel statusLabel;

    private JButton executeAgainButton;
    private JButton stopButton;
    private JButton findButton;
    private JToggleButton pinButton;
    private JButton addToFavoritesButton;

    private JPanel linkNavigationPanel;

    private JLabel forceShowOutputButtonLabel;   //a label that acts like a button

    private SearchPanel searchPanel;

    private boolean isBusy;     //is this actively showing output?
    private boolean isPending;  //is this waiting to show output?
    private boolean isPinned;   //keeps this panel open and disallows it from being re-used.
    private boolean showProgress = true;
    private boolean onlyShowOutputOnErrors;
    private boolean wasStopped;   //whether or not execution has been stopped by the user

    private Request request;
    private JButton nextLinkButton;
    private JButton previousLinkButton;

    public interface OutputPanelParent {

        public void removeOutputPanel(OutputPanel outputPanel);

        void reportExecuteFinished(Request request, boolean wasSuccessful);

        void executeAgain(Request request, OutputPanel outputPanel);

        public FileLinkDefinitionLord getFileLinkDefinitionLord();
    }

    public OutputPanel(GradlePluginLord gradlePluginLord, OutputPanelParent parent, AlternateUIInteraction alternateUIInteraction) {
        this.gradlePluginLord = gradlePluginLord;
        this.parent = parent;
        this.alternateUIInteraction = alternateUIInteraction;
    }

    /**
     * Call this after initializing this, but after setting any additional swing properties (actually, just the font for now). I really only added this as an optimization. Since we'll always be setting
     * the font, I didn't want the various style objects created only to be thrown away and re-created. This way, you can set the font before we create the styles.
     */
    public void initialize() {
        setupUI();
    }

    /**
     * This is called whenever a new request is made. It associates this request with this output panel.
     */
    public void setRequest(Request request, boolean onlyShowOutputOnErrors) {
        this.request = request;
        if (request.forceOutputToBeShown()) {
            setOnlyShowOutputOnErrors(false);
        } else {
            setOnlyShowOutputOnErrors(onlyShowOutputOnErrors);
        }

        enableAddToFavoritesAppropriately();

        //set this to indeterminate until we figure out how many tasks to execute.
        progressBar.setIndeterminate(true);
        progressBar.setStringPainted(false); //And don't show '0%' in the mean time.

        setPending(true);
        showProgress(true);   //make sure the progress is shown. It may have been turned off if we're reusing this component

        appendGradleOutput(getPrefixText());
    }

    /**
     * Returns a string stating the command we're currently executing. This is placed at the beginning of the output text. This is called when we start and when the command is finished (where we
     * replace all of our text with the total output)
     */
    private String getPrefixText() {
        return "Executing command: \"" + request.getFullCommandLine() + "\"\n";
    }

    public boolean isPinned() {
        return isPinned;
    }

    public void setPinned(boolean pinned) {
        isPinned = pinned;
        pinButton.setSelected(isPinned);
    }

    public boolean isBusy() {
        return isBusy;
    }

    protected void setBusy(boolean busy) {
        isBusy = busy;
    }   //this should be the only way to isBusy.

    public boolean isPending() {
        return isPending;
    }

    private void setPending(boolean pending) {
        isPending = pending;
        if (isPending) {
            statusLabel.setText("Waiting to execute");
        }

        progressBar.setVisible(!isPending);
    }

    public Request getRequest() {
        return request;
    }

    private void setupUI() {
        setLayout(new BorderLayout());

        add(createSideOptionsPanel(), BorderLayout.WEST);

        //why am I adding this center panel? Its so I can make the WEST side always stay in place. NORTH takes presedent over WEST normally.
        //and the NORTH panel here changes it height (when the progress bar comes and goes). This made the buttons along the WEST layout move
        //and was very jarring if you were about to click one. Now the buttons stay in place.
        JPanel centerPanel = new JPanel(new BorderLayout());
        add(centerPanel, BorderLayout.CENTER);

        centerPanel.add(createGradleOutputPanel(), BorderLayout.CENTER);
        centerPanel.add(createInfoPanel(), BorderLayout.NORTH);
        centerPanel.add(createSearchPanel(), BorderLayout.SOUTH);
    }

    private Component createGradleOutputPanel() {
        gradleOutputTextPanel = new JPanel(new BorderLayout());

        gradleOutputTextPane = new OutputTextPane(new OutputTextPane.Interaction() {
            public void fileClicked(File file, int line) {
                alternateUIInteraction.openFile(file, line);
            }
        }, alternateUIInteraction.doesSupportEditingOpeningFiles(), getFont(), parent.getFileLinkDefinitionLord());

        gradleOutputTextPanel.add(gradleOutputTextPane.asComponent(), BorderLayout.CENTER);

        return gradleOutputTextPanel;
    }

    private Component createInfoPanel() {
        JPanel panel = new JPanel();
        panel.setLayout(new BoxLayout(panel, BoxLayout.Y_AXIS));

        panel.add(createStatusPanel());
        panel.add(createProgressPanel());

        return panel;
    }

    private Component createProgressPanel() {
        progressPanel = new JPanel(new BorderLayout());
        progressLabel = new JLabel("Progress");
        progressBar = new JProgressBar();
        progressBar.setStringPainted(true);

        progressPanel.add(progressBar, BorderLayout.NORTH);
        progressPanel.add(progressLabel, BorderLayout.SOUTH);

        progressPanel.setBorder(BorderFactory.createEmptyBorder(2, 2, 2, 2));

        progressPanel.setVisible(false);
        return progressPanel;
    }

    private Component createStatusPanel() {
        statusPanel = new JPanel();
        statusPanel.setLayout(new BoxLayout(statusPanel, BoxLayout.X_AXIS));
        statusLabel = new JLabel();

        //this button is only shown when the output is hidden
        forceShowOutputButtonLabel = new JLabel("Show Output");

        forceShowOutputButtonLabel.addMouseListener(new MouseAdapter() {
            public void mouseClicked(MouseEvent e) {
                forciblyShowOutput();
            }

            public void mouseEntered(MouseEvent e) {
                forceShowOutputButtonLabel.setForeground(UIManager.getColor("textHighlightText"));
            }

            public void mouseExited(MouseEvent e) {
                forceShowOutputButtonLabel.setForeground(UIManager.getColor("Label.foreground"));
            }
        });
        statusPanel.add(statusLabel);
        statusPanel.add(Box.createHorizontalGlue());
        statusPanel.add(forceShowOutputButtonLabel);

        statusPanel.setBorder(BorderFactory.createEmptyBorder(2, 2, 2, 2));
        return statusPanel;
    }

    /**
     * This creates a panel that has several options such as execute again, stop, cancel, go to next link, etc..
     */
    private Component createSideOptionsPanel() {
        executeAgainButton = Utility.createButton(OutputPanel.class, "/org/gradle/gradleplugin/userinterface/swing/generic/tabs/execute.png", "Execute again", new AbstractAction() {
            public void actionPerformed(ActionEvent e) {
                parent.executeAgain(request, OutputPanel.this);
            }
        });

        executeAgainButton.setEnabled(false);

        stopButton = Utility.createButton(OutputPanel.class, "/org/gradle/gradleplugin/userinterface/swing/generic/stop.png", "Stop executing", new AbstractAction() {
            public void actionPerformed(ActionEvent e) {
                stop();
            }
        });
        stopButton.setEnabled(true);

        findButton = Utility.createButton(OutputPanel.class, "/org/gradle/gradleplugin/userinterface/swing/generic/find.png", "Find in output", new AbstractAction() {
            public void actionPerformed(ActionEvent e) {
                searchPanel.show();
            }
        });

        pinButton = Utility.createToggleButton(OutputPanel.class, "/org/gradle/gradleplugin/userinterface/swing/generic/pin.png", "Pin this output tab", new AbstractAction() {
            public void actionPerformed(ActionEvent e) {
                setPinned(!isPinned);
            }
        });

        addToFavoritesButton = Utility.createButton(OutputPanel.class, "/org/gradle/gradleplugin/userinterface/swing/generic/add-favorite.png", "Add to favorites", new AbstractAction() {
            public void actionPerformed(ActionEvent e) {
                addToFavorites();
            }
        });


        JPanel panel = new JPanel();
        panel.setLayout(new BoxLayout(panel, BoxLayout.Y_AXIS));

        panel.setBorder(BorderFactory.createEmptyBorder(2, 2, 2, 2)); //not only does this make it look better, it's really need so the pin toggle shows up clearly.

        panel.add(executeAgainButton);
        panel.add(Box.createVerticalStrut(5));
        panel.add(stopButton);
        panel.add(Box.createVerticalStrut(10));
        panel.add(pinButton);
        panel.add(Box.createVerticalStrut(10));
        panel.add(findButton);
        panel.add(Box.createVerticalStrut(10));
        panel.add(createLinkNavigationOptions());
        //the navigation options create a vertical strut and is only shown if the options are present, so we don't need to add one here.
        panel.add(addToFavoritesButton);

        panel.add(Box.createVerticalGlue());

        return panel;
    }

    /**
     * This creates a panel that has options for going to the next and previous link. This may be entirely hidden if the user doesn't have the ability to open file links.
     */
    private Component createLinkNavigationOptions() {

        linkNavigationPanel = new JPanel();
        linkNavigationPanel.setLayout(new BoxLayout(linkNavigationPanel, BoxLayout.Y_AXIS));

        nextLinkButton = Utility.createButton(getClass(), "/org/gradle/gradleplugin/userinterface/swing/generic/next-link.png", "Go to the next link", new AbstractAction() {
            public void actionPerformed(ActionEvent e) {
                goToNextLink();
            }
        });

        previousLinkButton = Utility.createButton(getClass(), "/org/gradle/gradleplugin/userinterface/swing/generic/previous-link.png", "Go to the previous link", new AbstractAction() {
            public void actionPerformed(ActionEvent e) {
                goToPreviousLink();
            }
        });

        linkNavigationPanel.add(previousLinkButton);
        linkNavigationPanel.add(Box.createVerticalStrut(5));
        linkNavigationPanel.add(nextLinkButton);
        linkNavigationPanel.add(Box.createVerticalStrut(10));

        if (!alternateUIInteraction.doesSupportEditingOpeningFiles()) {
            linkNavigationPanel.setVisible(false); //If we don't support it, hide this panel. Its just easier to create the above controls and know they're always present and hide them than constantly check for nulls.
        }

        return linkNavigationPanel;
    }

    private Component createSearchPanel() {
        StyleContext styleContent = StyleContext.getDefaultStyleContext();

        AttributeSet highlightStyle = gradleOutputTextPane.getDefaultStyle().copyAttributes();
        highlightStyle = styleContent.addAttribute(highlightStyle, StyleConstants.Foreground, Color.white);
        highlightStyle = styleContent.addAttribute(highlightStyle, StyleConstants.Background, Color.orange);
        highlightStyle = styleContent.addAttribute(highlightStyle, StyleConstants.Underline, true);

        AttributeSet emphasizedHighlightStyle = highlightStyle.copyAttributes();
        emphasizedHighlightStyle = styleContent.addAttribute(emphasizedHighlightStyle, StyleConstants.Foreground, Color.black);
        emphasizedHighlightStyle = styleContent.addAttribute(emphasizedHighlightStyle, StyleConstants.Background, Color.yellow);

        searchPanel = new SearchPanel(new OutputPanelSearchInteraction(gradleOutputTextPane.getTextComponent(), gradleOutputTextPane.getDefaultStyle(), highlightStyle, emphasizedHighlightStyle));
        searchPanel.hide();

        return searchPanel.getComponent();
    }

    /**
     * Special implementation just so can control how results are repainted. Specifically, so we can erase search highlights.
     */
    private class OutputPanelSearchInteraction extends TextPaneSearchInteraction {
        private OutputPanelSearchInteraction(JTextPane textComponentToSearch, AttributeSet defaultStyle, AttributeSet highlightStyle, AttributeSet emphasizedHighlightStyle) {
            super(textComponentToSearch, defaultStyle, highlightStyle, emphasizedHighlightStyle);
        }

        /**
         * We override this so we can handle our more-complicated case. The base class will remove the highlighting, and in doing so, will remove ALL highlighting. We want to keep the highlighting we've
         * specially added.
         */
        @Override
        public void removeResultHighlights() {
            gradleOutputTextPane.resetHighlights();
        }
    }

    private void goToNextLink() {
        FileLink fileLink = gradleOutputTextPane.getNextFileLink();
        gradleOutputTextPane.selectFileLink(fileLink);
    }

    private void goToPreviousLink() {
        FileLink fileLink = gradleOutputTextPane.getPreviousFileLink();
        gradleOutputTextPane.selectFileLink(fileLink);
    }

    /**
     * Call this before you use this. It resets its output as well as enabling buttons appropriately.
     */
    public void reset() {
        executeAgainButton.setEnabled(false);
        stopButton.setEnabled(true);
        statusLabel.setText("");
        statusLabel.setForeground(UIManager.getColor("Label.foreground"));
        gradleOutputTextPane.setText("");
        progressLabel.setText("");
        wasStopped = false;

        searchPanel.hide();

        previousLinkButton.setEnabled(false);
        nextLinkButton.setEnabled(false);
    }

    /**
     * Call this to append text to the gradle output field. We'll also move the caret to the end.
     *
     * @param text the text to add
     */
    private void appendGradleOutput(final String text) {
        SwingUtilities.invokeLater(new Runnable() {
            public void run() {
                gradleOutputTextPane.appendText(text);
                updateLinkNavigationOptions();
            }
        });
    }

    private void setProgress(final String text, final float percentComplete) {
        SwingUtilities.invokeLater(new Runnable() {
            public void run() {
                progressBar.setValue((int) percentComplete);
                progressLabel.setText(text);
            }
        });
    }

    /**
     * Notification that execution of a task or tasks has been started.
     */
    public void reportExecutionStarted() {
        SwingUtilities.invokeLater(new Runnable() {
            public void run() {
                setPending(false);
                setBusy(true);
                setProgress("Starting", 0);
                if (showProgress) {
                    progressPanel.setVisible(true);
                }

                statusLabel.setText("Executing");

                //give the user the option to override this.
                forceShowOutputButtonLabel.setVisible(onlyShowOutputOnErrors);
            }
        });
    }

    /**
     * Notification of the total number of tasks that will be executed. This is called after reportExecutionStarted and before any tasks are executed.
     *
     * @param size the total number of tasks.
     */
    public void reportNumberOfTasksToExecute(final int size) {  //if we only have a single task, then the entire process will be indeterminately long (it'll just from 0 to 100)
        SwingUtilities.invokeLater(new Runnable() {
            public void run() {
                boolean isIndeterminate = size == 1;
                progressBar.setIndeterminate(isIndeterminate);
                progressBar.setStringPainted(!isIndeterminate);
            }
        });
    }

    /**
     * Notification that execution of all tasks has completed. This is only called once at the end.
     *
     * @param wasSuccessful whether or not gradle encountered errors.
     * @param buildResult contains more detailed information about the result of a build.
     * @param output the text that gradle produced. May contain error information, but is usually just status.
     */
    public void reportExecutionFinished(boolean wasSuccessful, BuildResult buildResult, String output) {
        reportExecutionFinished(wasSuccessful, output, buildResult.getFailure());
    }

    /**
     * Notification that execution of a task has completed. This is the task you initiated and not for each subtask or dependent task.
     *
     * @param wasSuccessful whether or not gradle encountered errors.
     * @param output the text that gradle produced. May contain error information, but is usually just status.
     */
    public void reportExecutionFinished(final boolean wasSuccessful, final String output, final Throwable throwable) {
        SwingUtilities.invokeLater(new Runnable() {
            public void run() {
                setPending(false); //this can be called before we actually get a start message if it fails early. This clears the pending flag so we know we can reuse it.
                setBusy(false);
                progressPanel.setVisible(false);

                //Make the output equal to all of our output. There are some timing issues where we don't get the last live output from gradle.
                //This 'output' is the entire text. This way we always get all output.
                String newText = getPrefixText() + output;
                gradleOutputTextPane.setText(newText);

                //show the user the time we finished this.
                SimpleDateFormat formatter = new SimpleDateFormat("h:mm:ss aa");
                String formattedTime = formatter.format(Calendar.getInstance().getTime());

                if (wasSuccessful) {
                    statusLabel.setText("Completed successfully at " + formattedTime);
                    appendGradleOutput("\nCompleted Successfully");
                } else {
                    if (wasStopped) {
                        statusLabel.setText("User stopped execution at " + formattedTime);
                    } else {
                        statusLabel.setText("Completed with errors at " + formattedTime);
                    }

                    statusLabel.setForeground(Color.red.darker());

                    //since errors occurred, show the output. If onlyShowOutputOnErrors is false, this textPanel will already be visible.
                    gradleOutputTextPanel.setVisible(true);
                }

                executeAgainButton.setEnabled(true);
                stopButton.setEnabled(false);

                appendThrowable(throwable);

                //lastly, if the text output is not visible, make the 'show output' button visible
                forceShowOutputButtonLabel.setVisible(!gradleOutputTextPanel.isVisible());
                updateLinkNavigationOptions();

                searchPanel.performSearchAgain(); //this will update our results if the user was searching during the execution

                parent.reportExecuteFinished(request, wasSuccessful);

            }
        });
    }

    private void appendThrowable(Throwable throwable) {
        if (throwable != null) {
            String output = GradlePluginLord.getGradleExceptionMessage(throwable, ShowStacktrace.ALWAYS_FULL);
            appendGradleOutput(output);
        }
    }

    /**
     * Notification that a single task has completed. Note: the task you kicked off probably executes other tasks.
     *
     * @param currentTaskName the task being executed
     * @param percentComplete the percent complete of all the tasks that make up the task you requested.
     */
    public void reportTaskStarted(String currentTaskName, float percentComplete) {
        setProgress(currentTaskName, percentComplete);
    }

    public void reportTaskComplete(String currentTaskName, float percentComplete) {
        setProgress(currentTaskName, percentComplete);
    }

    public void reportFatalError(String message) {
        appendGradleOutput('\n' + message + "\n\nFailed.\n");
    }

    /**
     * Report real-time output from gradle and its subsystems (such as ant).
     *
     * @param output a single line of text to show.
     */
    public void reportLiveOutput(String output) {
        appendGradleOutput(output);
    }

    /**
     * Determines if this panel is ready to be reused. Currently, if its not busy or pinned, it can be reused.
     *
     */
    public boolean canBeReusedNow() {
        return !isPending && !isBusy && !isPinned;
    }

    /**
     * Call this to show progress. Some tasks have no useful progress, so this allows you to disable it.
     *
     * @param showProgress true to show a progress bar, false not to.
     */
    private void showProgress(boolean showProgress) {
        this.showProgress = showProgress;
        progressPanel.setVisible(showProgress);
    }

    /**
     * This overrides the onlyShowOutputOnErrors
     */
    private void forciblyShowOutput() {
        gradleOutputTextPanel.setVisible(true);
        forceShowOutputButtonLabel.setVisible(false);
    }

    public void setOnlyShowOutputOnErrors(boolean value) {
        this.onlyShowOutputOnErrors = value;
        gradleOutputTextPanel.setVisible(!value);
    }

    public boolean getOnlyShowOutputOnErrors() {
        return onlyShowOutputOnErrors;
    }

    public boolean close() {

        if (!stop()) {
            return false;
        }

        parent.removeOutputPanel(this);

        setPinned(false);  //unpin it when it is removed
        return true;
    }

    /**
     * This stops the currently executing task if any exists.
     *
     * @return true if the request stopped, false if not.
     */
    public boolean stop() {
        if (request != null) { //if we have a request, we can only close if it allows us to.
            if (!request.cancel()) {
                return false;
            }
        }

        wasStopped = true;

        return true;
    }


    /**
     * Sets the font for this component.
     *
     * @param font the desired Font for this component
     * @beaninfo preferred: true bound: true attribute: visualUpdate true description: The font for the component.
     * @see Component#getFont
     */
    @Override
    public void setFont(Font font) {
        super.setFont(font);
        if (gradleOutputTextPane != null) { //this gets called by internal Swing APIs, so we may not have this yet.
            gradleOutputTextPane.setFont(font);
        }
    }

    /**
     * Shows or hides the link navigation options appropriately
     */
    private void updateLinkNavigationOptions() {
        if (gradleOutputTextPane.hasClickableLinks()) {

            nextLinkButton.setEnabled(true);
            previousLinkButton.setEnabled(true);
        } else {
            nextLinkButton.setEnabled(false);
            previousLinkButton.setEnabled(false);
        }
    }

    /**
     * Adds the current request to the favorites and allows the user to edit it.
     */
    private void addToFavorites() {
        if (request == null) {
            return;
        }

        String fullCommandLine = request.getFullCommandLine();
        String displayName = request.getDisplayName();

        FavoriteTask favoriteTask = gradlePluginLord.getFavoritesEditor().addFavorite(fullCommandLine, displayName, false);
        if (favoriteTask != null) {
            gradlePluginLord.getFavoritesEditor().editFavorite(favoriteTask, new SwingEditFavoriteInteraction(SwingUtilities.getWindowAncestor(this), "Edit Favorite", SwingEditFavoriteInteraction.SynchronizeType.OnlyIfAlreadySynchronized));
            enableAddToFavoritesAppropriately();
        }
    }

    /**
     * This shows or hides the 'add to favorites' button based on the current request.
     */
    private void enableAddToFavoritesAppropriately() {
        boolean isVisible = true;
        if (request == null) {
            isVisible = false;
        } else {
            //adding 'refresh' to favorites no sense. Hide this button in that case.
            if (request.getType() == RefreshTaskListRequest.TYPE) {
                isVisible = false;
            } else if (gradlePluginLord.getFavoritesEditor().getFavorite(request.getFullCommandLine()) != null) { //is it a command that's already a favorite?
                isVisible = false;
            }
        }

        addToFavoritesButton.setVisible(isVisible);
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy