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

com.palantir.javaformat.java.JavaCommentsHelper Maven / Gradle / Ivy

There is a newer version: 2.50.0
Show newest version
/*
 * Copyright 2015 Google Inc.
 *
 * 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 com.palantir.javaformat.java;

import com.google.common.base.CharMatcher;
import com.palantir.javaformat.CommentsHelper;
import com.palantir.javaformat.Input.Tok;
import com.palantir.javaformat.Newlines;
import com.palantir.javaformat.java.javadoc.JavadocFormatter;
import java.util.ArrayList;
import java.util.Iterator;
import java.util.List;
import java.util.regex.Matcher;
import java.util.regex.Pattern;

/** {@code JavaCommentsHelper} extends {@link CommentsHelper} to rewrite Java comments. */
public final class JavaCommentsHelper implements CommentsHelper {

    private final String lineSeparator;
    private final JavaFormatterOptions options;
    private final JavadocFormatter javadocFormatter;

    public JavaCommentsHelper(String lineSeparator, JavaFormatterOptions options) {
        this.lineSeparator = lineSeparator;
        this.options = options;
        this.javadocFormatter = options.formatJavadoc() ? new JavadocFormatter(options.maxLineLength()) : null;
    }

    @Override
    public String rewrite(Tok tok, int maxWidth, int column0) {
        if (!tok.isComment()) {
            return tok.getOriginalText();
        }
        String text = tok.getOriginalText();
        if (javadocFormatter != null && tok.isJavadocComment()) {
            text = javadocFormatter.formatJavadoc(text, column0);
        }
        List lines = new ArrayList<>();
        Iterator it = Newlines.lineIterator(text);
        while (it.hasNext()) {
            lines.add(CharMatcher.whitespace().trimTrailingFrom(it.next()));
        }
        if (tok.isSlashSlashComment()) {
            return indentLineComments(lines, column0);
        } else if (javadocShaped(lines)) {
            return indentJavadoc(lines, column0);
        } else {
            return preserveIndentation(lines, column0);
        }
    }

    // For non-javadoc-shaped block comments, shift the entire block to the correct
    // column, but do not adjust relative indentation.
    private String preserveIndentation(List lines, int column0) {
        StringBuilder builder = new StringBuilder();

        // find the leftmost non-whitespace character in all trailing lines
        int startCol = -1;
        for (int i = 1; i < lines.size(); i++) {
            int lineIdx = CharMatcher.whitespace().negate().indexIn(lines.get(i));
            if (lineIdx >= 0 && (startCol == -1 || lineIdx < startCol)) {
                startCol = lineIdx;
            }
        }

        // output the first line at the current column
        builder.append(lines.get(0));

        // output all trailing lines with plausible indentation
        for (int i = 1; i < lines.size(); ++i) {
            builder.append(lineSeparator).append(" ".repeat(column0));
            // check that startCol is valid index, e.g. for blank lines
            if (lines.get(i).length() >= startCol) {
                builder.append(lines.get(i).substring(startCol));
            } else {
                builder.append(lines.get(i));
            }
        }
        return builder.toString();
    }

    // Wraps and re-indents line comments.
    private String indentLineComments(List lines, int column0) {
        lines = wrapLineComments(lines, column0);
        StringBuilder builder = new StringBuilder();
        builder.append(lines.get(0).trim());
        String indentString = " ".repeat(column0);
        for (int i = 1; i < lines.size(); ++i) {
            builder.append(lineSeparator)
                    .append(indentString)
                    .append(lines.get(i).trim());
        }
        return builder.toString();
    }

    // Preserve special `//noinspection` and `//$NON-NLS-x$` comments used by IDEs, which cannot
    // contain leading spaces.
    private static final Pattern LINE_COMMENT_MISSING_SPACE_PREFIX =
            Pattern.compile("^(//+)(?!noinspection|\\$NON-NLS-\\d+\\$)[^\\s/]");

    private List wrapLineComments(List lines, int column0) {
        List result = new ArrayList<>();
        for (String line : lines) {
            // Add missing leading spaces to line comments: `//foo` -> `// foo`.
            Matcher matcher = LINE_COMMENT_MISSING_SPACE_PREFIX.matcher(line);
            if (matcher.find()) {
                int length = matcher.group(1).length();
                line = "/".repeat(length) + " " + line.substring(length);
            }
            if (line.startsWith("// MOE:")) {
                // don't wrap comments for https://github.com/google/MOE
                result.add(line);
                continue;
            }
            while (line.length() + column0 > options.maxLineLength()) {
                int idx = options.maxLineLength() - column0;
                // only break on whitespace characters, and ignore the leading `// `
                while (idx >= 2 && !CharMatcher.whitespace().matches(line.charAt(idx))) {
                    idx--;
                }
                if (idx <= 2) {
                    break;
                }
                result.add(line.substring(0, idx));
                line = "//" + line.substring(idx);
            }
            result.add(line);
        }
        return result;
    }

    // Remove leading whitespace (trailing was already removed), and re-indent.
    // Add a +1 indent before '*', and add the '*' if necessary.
    private String indentJavadoc(List lines, int column0) {
        StringBuilder builder = new StringBuilder();
        builder.append(lines.get(0).trim());
        int indent = column0 + 1;
        String indentString = " ".repeat(indent);
        for (int i = 1; i < lines.size(); ++i) {
            builder.append(lineSeparator).append(indentString);
            String line = lines.get(i).trim();
            if (!line.startsWith("*")) {
                builder.append("* ");
            }
            builder.append(line);
        }
        return builder.toString();
    }

    // Returns true if the comment looks like javadoc
    private static boolean javadocShaped(List lines) {
        Iterator it = lines.iterator();
        if (!it.hasNext()) {
            return false;
        }
        String first = it.next().trim();
        // if it's actually javadoc, we're done
        if (first.startsWith("/**")) {
            return true;
        }
        // if it's a block comment, check all trailing lines for '*'
        if (!first.startsWith("/*")) {
            return false;
        }
        while (it.hasNext()) {
            if (!it.next().trim().startsWith("*")) {
                return false;
            }
        }
        return true;
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy