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

com.palantir.javaformat.doc.DocBuilder Maven / Gradle / Ivy

/*
 * (c) Copyright 2019 Palantir Technologies Inc. All rights reserved.
 *
 * 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.doc;

import com.google.common.base.MoreObjects;
import com.palantir.javaformat.Indent;
import com.palantir.javaformat.Op;
import com.palantir.javaformat.OpenOp;
import com.palantir.javaformat.OpsBuilder;
import java.util.ArrayDeque;
import java.util.List;

/** A {@code DocBuilder} converts a sequence of {@link Op}s into a {@link Doc}. */
public final class DocBuilder {
    private final Level base = Level.make(
            OpenOp.builder().plusIndent(Indent.Const.ZERO).debugName("root").build());
    private final ArrayDeque stack = new ArrayDeque<>();

    /**
     * A possibly earlier {@link Level} for appending text, à la Philip Wadler.
     *
     * 

Processing {@link Doc}s presents a subtle problem. Suppose we have a {@link Doc} for to an assignment node, * {@code a = b}, with an optional {@link Break} following the {@code =}. Suppose we have 5 characters to write it, * so that we think we don't need the break. Unfortunately, this {@link Doc} lies in an expression statement * {@link Doc} for the statement {@code a = b;} and this statement does not fit in 3 characters. This is why many * formatters sometimes emit lines that are too long, or cheat by using a narrower line length to avoid such * problems. * *

One solution to this problem is not to decide whether a {@link Level} should be broken until later (in this * case, after the semicolon has been seen). A simpler approach is to rewrite the {@link Doc} as here, so that the * semicolon moves inside the inner {@link Doc}, and we can decide whether to break that {@link Doc} without seeing * later text. */ private Level appendLevel = base; /** Start to build a {@code DocBuilder}. */ public DocBuilder() { stack.addLast(base); } /** * Add a list of {@link Op}s to the {@link OpsBuilder}. * * @param ops the {@link Op}s * @return the {@link OpsBuilder} */ public DocBuilder withOps(List ops) { for (Op op : ops) { op.add(this); // These operations call the operations below to build the doc. } return this; } /** Open a new {@link Level}. */ public void open(OpenOp openOp) { Level level = Level.make(openOp); stack.addLast(level); } /** Close the current {@link Level}. */ public void close() { Level top = stack.removeLast(); stack.peekLast().add(top); } /** * Add a {@link Doc} to the current {@link Level}. * * @param doc the {@link Doc} */ void add(Doc doc) { appendLevel.add(doc); } /** * Add a {@link Break} to the current {@link Level}. * * @param breakDoc the {@link Break} */ void breakDoc(Break breakDoc) { appendLevel = stack.peekLast(); appendLevel.add(breakDoc); } /** * Return the {@link Doc}. * * @return the {@link Doc} */ public Level build() { return base; } @Override public String toString() { return MoreObjects.toStringHelper(this) .add("base", base) .add("stack", stack) .add("appendLevel", appendLevel) .toString(); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy