com.palantir.javaformat.doc.Level Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of palantir-java-format Show documentation
Show all versions of palantir-java-format Show documentation
Palantir open source project
/*
* (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 static com.google.common.collect.Iterables.getLast;
import com.google.common.base.MoreObjects;
import com.google.common.base.Preconditions;
import com.google.common.base.Suppliers;
import com.google.common.collect.ImmutableList;
import com.google.common.collect.Range;
import com.palantir.javaformat.BreakBehaviour;
import com.palantir.javaformat.BreakBehaviours;
import com.palantir.javaformat.CommentsHelper;
import com.palantir.javaformat.Indent;
import com.palantir.javaformat.LastLevelBreakability;
import com.palantir.javaformat.OpenOp;
import com.palantir.javaformat.OpenOp.Complexity;
import com.palantir.javaformat.Output;
import com.palantir.javaformat.PartialInlineability;
import com.palantir.javaformat.doc.Obs.Exploration;
import com.palantir.javaformat.doc.Obs.ExplorationNode;
import com.palantir.javaformat.doc.Obs.LevelNode;
import com.palantir.javaformat.doc.StartsWithBreakVisitor.Result;
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;
import java.util.ArrayList;
import java.util.List;
import java.util.Optional;
import java.util.OptionalInt;
import java.util.stream.Collector;
import java.util.stream.Collectors;
import java.util.stream.Stream;
import org.immutables.value.Value;
/** A {@code Level} inside a {@link Doc}. */
public final class Level extends Doc {
/**
* How many branches we are allowed to take (i.e how many times we can consider breaking vs not breaking the current
* level) before we stop branching and always break, which is the google-java-format default behaviour.
*/
private static final int MAX_BRANCHING_COEFFICIENT = 20;
private static final Collector> GET_LAST_COLLECTOR = Collectors.reducing((u, v) -> v);
private final List docs = new ArrayList<>(); // The elements of the level.
@SuppressWarnings("Immutable") // Effectively immutable
private final ImmutableSupplier memoizedSplitsBreaks =
Suppliers.memoize(() -> splitByBreaks(docs))::get;
/** The immutable characteristics of this level determined before the level contents are available. */
private final OpenOp openOp;
private Level(OpenOp openOp) {
this.openOp = openOp;
}
/** Factory method for {@code Level}s. */
static Level make(OpenOp openOp) {
return new Level(openOp);
}
/**
* Add a {@link Doc} to the {@code Level}.
*
* @param doc the {@link Doc} to add
*/
void add(Doc doc) {
docs.add(doc);
}
@Override
protected float computeWidth() {
float thisWidth = 0.0F;
for (Doc doc : docs) {
thisWidth += doc.getWidth();
}
return thisWidth;
}
@Override
protected String computeFlat() {
StringBuilder builder = new StringBuilder();
for (Doc doc : docs) {
builder.append(doc.getFlat());
}
return builder.toString();
}
@Override
protected Range computeRange() {
Range docRange = EMPTY_RANGE;
for (Doc doc : docs) {
docRange = union(docRange, doc.range());
}
return docRange;
}
@Override
public State computeBreaks(CommentsHelper commentsHelper, int maxWidth, State state, Obs.ExplorationNode observer) {
return tryToFitOnOneLine(maxWidth, state, docs)
.map(newWidth -> state.withColumn(newWidth).withLevelState(this, ImmutableLevelState.of(true)))
.orElseGet(() -> {
Obs.LevelNode childLevel = observer.newChildNode(this, state);
State newState =
getBreakBehaviour().match(new BreakImpl(commentsHelper, maxWidth, state, childLevel));
return childLevel.finishLevel(state.updateAfterLevel(newState));
});
}
/**
* Try to fit these docs belonging to the current level onto one line, returning empty if we couldn't. This takes
* into account the level's {@link #getColumnLimitBeforeLastBreak()}.
*
* @return the width after fitting it onto one line, if it was possible. This is guaranteed to be less than
* {@code maxWidth}
*/
private Optional tryToFitOnOneLine(int maxWidth, State state, Iterable docs) {
int column = state.column();
int columnBeforeLastBreak = 0; // Not activated by default
for (Doc doc : docs) {
if (doc instanceof Break && ((Break) doc).hasColumnLimit()) {
columnBeforeLastBreak = column;
} else if (doc instanceof Level) {
// Levels might have nested levels that have a 'columnLimitBeforeLastBreak' set, so recurse.
State newState = state.withColumn(column);
Level innerLevel = (Level) doc;
Optional newWidth = innerLevel.tryToFitOnOneLine(maxWidth, newState, innerLevel.getDocs());
if (!newWidth.isPresent()) {
return Optional.empty();
}
column = newWidth.get();
continue;
}
column = (int) (column + doc.getWidth());
}
// Make an additional check that widthBeforeLastBreak fits in the column limit
if (getColumnLimitBeforeLastBreak().isPresent()
&& columnBeforeLastBreak > getColumnLimitBeforeLastBreak().getAsInt()) {
return Optional.empty();
}
// Check that the entirety of this level fits on the current line.
if (column <= maxWidth) {
return Optional.of(column);
}
return Optional.empty();
}
class BreakImpl implements BreakBehaviour.Cases {
private final CommentsHelper commentsHelper;
private final int maxWidth;
private final State state;
private final Obs.LevelNode levelNode;
public BreakImpl(CommentsHelper commentsHelper, int maxWidth, State state, Obs.LevelNode levelNode) {
this.commentsHelper = commentsHelper;
this.maxWidth = maxWidth;
this.state = state;
this.levelNode = levelNode;
}
private Exploration breakNormally(State state) {
return Level.this.breakNormally(state, levelNode, commentsHelper, maxWidth);
}
@Override
public State breakThisLevel() {
return breakNormally(state).markAccepted();
}
@Override
public State preferBreakingLastInnerLevel(boolean _keepIndentWhenInlined) {
// Try both breaking and not breaking. Choose the better one based on LOC, preferring
// breaks if the outcome is the same.
State state = this.state.withNewBranch();
Obs.Exploration broken = breakNormally(state);
if (state.branchingCoefficient() < MAX_BRANCHING_COEFFICIENT) {
State state1 = state.withNoIndent();
Optional lastLevelBroken = levelNode.maybeExplore(
"tryBreakLastLevel",
state1,
explorationNode -> tryBreakLastLevel(commentsHelper, maxWidth, state1, explorationNode, true));
if (lastLevelBroken.isPresent()) {
if (lastLevelBroken.get().state().numLines()
< broken.state().numLines()) {
return lastLevelBroken.get().markAccepted();
}
}
}
return broken.markAccepted();
}
@Override
public State breakOnlyIfInnerLevelsThenFitOnOneLine(boolean keepIndentWhenInlined) {
Exploration broken = Level.this.breakNormally(state, levelNode, commentsHelper, maxWidth);
Optional maybeInlined = levelNode.maybeExplore(
"handle_breakOnlyIfInnerLevelsThenFitOnOneLine",
state,
explorationNode -> handle_breakOnlyIfInnerLevelsThenFitOnOneLine(
commentsHelper, maxWidth, state, broken.state(), keepIndentWhenInlined, explorationNode));
if (maybeInlined.isPresent()) {
return maybeInlined.get().markAccepted();
} else {
return broken.markAccepted();
}
}
@Override
public State inlineSuffix() {
Optional lastLevelBroken = levelNode.maybeExplore(
"inlineSuffix",
state,
explorationNode -> tryInlineSuffix(commentsHelper, maxWidth, state, explorationNode, true));
return lastLevelBroken.orElseGet(() -> this.breakNormally(state)).markAccepted();
}
}
private Exploration breakNormally(State state, LevelNode levelNode, CommentsHelper commentsHelper, int maxWidth) {
State stateForBroken = state.withIndentIncrementedBy(getPlusIndent());
return levelNode.explore(
"breaking normally",
stateForBroken,
explorationNode -> computeBroken(commentsHelper, maxWidth, stateForBroken, explorationNode));
}
/**
* Attempts to perform the {@link BreakBehaviour.Cases#breakOnlyIfInnerLevelsThenFitOnOneLine} logic, returning
* empty if it couldn't. Namely, this will only return a state if:
*
* - everything fit on the current or on the next line (we verify this by checking that none of the inner
* levels were broken; or failing that
*
- a sensible prefix of this level fits onto the current line. See {@link CountWidthUntilBreakVisitor} for
* how we determine the full prefix that must fit on the current line.
*
*
* @param brokenState is expected to be the state resulting from {@link Level#breakNormally}, but is passed in
* for efficiency reasons
*/
private Optional handle_breakOnlyIfInnerLevelsThenFitOnOneLine(
CommentsHelper commentsHelper,
int maxWidth,
State state,
State brokenState,
boolean keepIndent,
Obs.ExplorationNode explorationNode) {
// Note: we are not checking if the brokenState produced one extra line compared to state, as this can be
// misleading if there is no level but a single comment that got reflowed onto multiple lines (see palantir-11).
boolean anyLevelWasBroken = getNonEmptyInnerLevels().anyMatch(level -> !brokenState.isOneLine(level));
if (!anyLevelWasBroken) {
return Optional.of(brokenState);
}
Optional partiallyInlinedStateOpt =
tryInlinePrefixOntoCurrentLine(commentsHelper, maxWidth, state, keepIndent, explorationNode);
if (!partiallyInlinedStateOpt.isPresent()) {
return Optional.empty();
}
State partiallyInlinedState = partiallyInlinedStateOpt.get();
boolean bodyIsComplex = getNonEmptyInnerLevels().anyMatch(il -> il.openOp.complexity() == Complexity.COMPLEX);
if (bodyIsComplex || partiallyInlinedState.numLines() < brokenState.numLines()) {
return Optional.of(partiallyInlinedState);
}
return Optional.empty();
}
private Stream getNonEmptyInnerLevels() {
return docs.stream()
.filter(doc -> doc instanceof Level)
.map(doc -> ((Level) doc))
.filter(doc -> StartsWithBreakVisitor.INSTANCE.visit(doc) != Result.EMPTY);
}
private Optional tryInlinePrefixOntoCurrentLine(
CommentsHelper commentsHelper,
int maxWidth,
State state,
boolean keepIndent,
Obs.ExplorationNode explorationNode) {
// Find the last level, skipping empty levels (that contain nothing, or are made up
// entirely of other empty levels).
// Last level because there might be other in-between levels after the initial break like `new
// int[] {`, and we want to skip those.
Level lastLevel = getNonEmptyInnerLevels()
.filter(doc -> StartsWithBreakVisitor.INSTANCE.visit(doc) != Result.EMPTY)
.collect(GET_LAST_COLLECTOR)
.orElseThrow(() ->
new IllegalStateException("Levels were broken so expected to find at least a non-empty level"));
// Add the width of tokens, breaks before the lastLevel. We must always have space for
// these.
List leadingDocs = docs.subList(0, docs.indexOf(lastLevel));
float leadingWidth = getWidth(leadingDocs);
// Potentially add the width of prefixes we want to consider as part of the width that
// must fit on the same line, so that we don't accidentally break prefixes when we could
// have avoided doing so.
leadingWidth += new CountWidthUntilBreakVisitor(maxWidth - state.indent()).visit(lastLevel);
boolean fits = !Float.isInfinite(leadingWidth) && state.column() + leadingWidth <= maxWidth;
if (!fits) {
return Optional.empty();
}
State newState = state.withNoIndent();
if (keepIndent) {
newState = newState.withIndentIncrementedBy(getPlusIndent());
}
return Optional.of(tryToLayOutLevelOnOneLine(
commentsHelper, maxWidth, newState, memoizedSplitsBreaks.get(), explorationNode));
}
private Optional tryBreakLastLevel(
CommentsHelper commentsHelper,
int maxWidth,
State state,
ExplorationNode explorationNode,
boolean isSimpleInliningSoFar) {
if (docs.isEmpty() || !(getLast(docs) instanceof Level)) {
return Optional.empty();
}
Level innerLevel = ((Level) getLast(docs));
return tryBreakInnerLevel(commentsHelper, maxWidth, state, explorationNode, innerLevel, isSimpleInliningSoFar);
}
private Optional tryInlineSuffix(
CommentsHelper commentsHelper,
int maxWidth,
State state,
ExplorationNode explorationNode,
boolean isSimpleInliningSoFar) {
Preconditions.checkState(
docs.size() > 0 && getLast(docs) instanceof Level,
"Level with break behaviour inlineSuffix must end with a level as its last doc");
Level lastLevel = (Level) getLast(docs);
Preconditions.checkState(
StartsWithBreakVisitor.INSTANCE.visit(lastLevel) == Result.YES,
"Level with break behaviour inlineSuffix must have a last level that starts with a break");
if (docs.size() < 2 || !(docs.get(docs.size() - 2) instanceof Level)) {
return Optional.empty();
}
Level lastLevelBeforeSuffix = (Level) docs.get(docs.size() - 2);
return tryBreakInnerLevel(
commentsHelper, maxWidth, state, explorationNode, lastLevelBeforeSuffix, isSimpleInliningSoFar);
}
private Optional tryBreakInnerLevel(
CommentsHelper commentsHelper,
int maxWidth,
State state,
ExplorationNode explorationNode,
Level innerLevel,
boolean isSimpleInliningSoFar) {
// Only split levels that have declared they want to be split in this way.
if (innerLevel.getBreakabilityIfLastLevel() == LastLevelBreakability.ABORT) {
return Optional.empty();
}
// See if we can fill in everything but the innerLevel.
// This is essentially like a small part of computeBreaks.
int innerLevelIndex = docs.indexOf(innerLevel);
List leadingDocs = docs.subList(0, innerLevelIndex);
if (!tryToFitOnOneLine(maxWidth, state, leadingDocs).isPresent()) {
return Optional.empty();
}
List trailingDocs = docs.subList(innerLevelIndex + 1, docs.size());
float trailingWidth = getWidth(trailingDocs);
if (Double.isInfinite(trailingWidth)) {
return Optional.empty();
}
// Note: we can't use computeBroken with 'leadingDocs' instead of 'docs', because it assumes
// _we_ are breaking.
// See computeBreakAndSplit -> shouldBreak
SplitsBreaks prefixSplitsBreaks = splitByBreaks(leadingDocs);
boolean isSimpleInlining = isSimpleInliningSoFar && Level.this.openOp.complexity() == Complexity.SIMPLE;
State state1 = tryToLayOutLevelOnOneLine(commentsHelper, maxWidth, state, prefixSplitsBreaks, explorationNode);
// If a break was still forced somehow even though we could fit the leadingWidth, then abort.
// This could happen if inner levels have set a `columnLimitBeforeLastBreak` or something like that.
if (state1.numLines() != state.numLines()) {
return Optional.empty();
}
Optional state2;
switch (innerLevel.getBreakabilityIfLastLevel()) {
case ACCEPT_INLINE_CHAIN_IF_SIMPLE_OTHERWISE_CHECK_INNER:
if (isSimpleInlining) {
state2 = tryBreakInnerLevel_acceptInlineChain(
commentsHelper, maxWidth, explorationNode, innerLevel, state1);
break;
}
// Otherwise, fall through to CHECK_INNER.
case CHECK_INNER:
state2 = tryBreakInnerLevel_checkInner(
commentsHelper, maxWidth, explorationNode, innerLevel, isSimpleInlining, state1);
break;
case ACCEPT_INLINE_CHAIN:
state2 = tryBreakInnerLevel_acceptInlineChain(
commentsHelper, maxWidth, explorationNode, innerLevel, state1);
break;
case ABORT:
state2 = Optional.empty();
break;
default:
throw new RuntimeException(
"Unexpected lastLevelBreakability: " + innerLevel.getBreakabilityIfLastLevel());
}
return state2.flatMap(stateAfterInner -> {
// Do we have a suffix to inline too?
return tryToFitOnOneLine(maxWidth, stateAfterInner, trailingDocs).map(stateAfterInner::withColumn);
});
}
private static Optional tryBreakInnerLevel_acceptInlineChain(
CommentsHelper commentsHelper,
int maxWidth,
ExplorationNode explorationNode,
Level lastLevel,
State state) {
// Ok then, we are allowed to break here, but first verify that we have enough room to inline this last
// level's prefix.
float extraWidth = new CountWidthUntilBreakVisitor(maxWidth - state.indent()).visit(lastLevel);
boolean stillFits = !Float.isInfinite(extraWidth) && state.column() + extraWidth <= maxWidth;
if (!stillFits) {
return Optional.empty();
}
// Note: computeBreaks, not computeBroken, so it can try to do this logic recursively for the
// lastLevel
return Optional.of(explorationNode
.newChildNode(lastLevel, state)
.explore(
"end tryBreakLastLevel chain",
state,
exp -> lastLevel.computeBreaks(commentsHelper, maxWidth, state, exp))
.markAccepted());
}
private static Optional tryBreakInnerLevel_checkInner(
CommentsHelper commentsHelper,
int maxWidth,
ExplorationNode explorationNode,
Level innerLevel,
boolean isSimpleInlining,
State state) {
// Try to fit the entire inner prefix if it's that kind of level.
return BreakBehaviours.caseOf(innerLevel.getBreakBehaviour())
.preferBreakingLastInnerLevel(keepIndentWhenInlined -> {
State state1 =
keepIndentWhenInlined ? state.withIndentIncrementedBy(innerLevel.getPlusIndent()) : state;
return explorationNode
.newChildNode(innerLevel, state1)
.maybeExplore(
"recurse into inner tryBreakLastLevel",
state1,
exp -> innerLevel.tryBreakLastLevel(
commentsHelper, maxWidth, state1, exp, isSimpleInlining))
.map(Exploration::markAccepted);
})
.inlineSuffix(() -> explorationNode
.newChildNode(innerLevel, state)
.maybeExplore(
"recurse into inner tryInlineSuffix",
state,
exp -> innerLevel.tryInlineSuffix(
commentsHelper, maxWidth, state, exp, isSimpleInlining))
.map(Exploration::markAccepted))
.breakOnlyIfInnerLevelsThenFitOnOneLine(keepIndentWhenInlined -> {
// This case currently only matches lambda _expressions_ (without curlies)
State state1 =
keepIndentWhenInlined ? state.withIndentIncrementedBy(innerLevel.getPlusIndent()) : state;
String humanDescription = "end tryBreakLastLevel chain -> breakOnlyIfInnerLevelsThenFitOnOneLine";
LevelNode levelNode = explorationNode.newChildNode(innerLevel, state1);
return levelNode
.maybeExplore(humanDescription, state1, exp -> {
// Not all levels would look good if inlined in this position, so we accept
// levels that are meant to look good even if partially inlined, e.g. method
// chains, which will catch things like builders, but not other kinds of levels like
// constant expressions.
// See the palantir-expression-lambdas.input test for an example of what this is
// trying to avoid.
// For this, need to actually check the last inner level of `lastLevel` (2 levels down).
if (innerLevel.docs.isEmpty() || !(getLast(innerLevel.docs) instanceof Level)) {
return Optional.empty();
}
Level lastLevel2 = ((Level) getLast(innerLevel.docs));
switch (lastLevel2.getBreakabilityIfLastLevel()) {
case ABORT:
case CHECK_INNER:
return Optional.empty();
case ACCEPT_INLINE_CHAIN:
Exploration broken =
innerLevel.breakNormally(state, levelNode, commentsHelper, maxWidth);
return innerLevel.handle_breakOnlyIfInnerLevelsThenFitOnOneLine(
commentsHelper,
maxWidth,
state1,
broken.state(),
keepIndentWhenInlined,
explorationNode);
case ACCEPT_INLINE_CHAIN_IF_SIMPLE_OTHERWISE_CHECK_INNER:
// We will allow inlining this kind of level but only if the level itself is
// not simple.
if (lastLevel2.openOp.complexity() == Complexity.SIMPLE) {
return Optional.empty();
}
return innerLevel.tryInlinePrefixOntoCurrentLine(
commentsHelper,
maxWidth,
state1,
keepIndentWhenInlined,
explorationNode);
default:
throw new RuntimeException("Unknown breakabilityIfLastLevel: " + lastLevel2);
}
})
.map(Exploration::markAccepted);
})
// We don't know how to fit the inner level on the same line, so bail out.
.otherwise_(Optional.empty());
}
/**
* Mark breaks in this level as not broken, but lay out the inner levels normally, according to their own
* {@link BreakBehaviour}. The resulting {@link State#mustBreak} will be true if this level did not fit on exactly
* one line.
*/
private State tryToLayOutLevelOnOneLine(
CommentsHelper commentsHelper,
int maxWidth,
State state,
SplitsBreaks splitsBreaks,
Obs.ExplorationNode explorationNode) {
for (int i = 0; i < splitsBreaks.splits().size(); ++i) {
if (i > 0) {
state = splitsBreaks.breaks().get(i - 1).computeBreaks(state, false);
}
List split = splitsBreaks.splits().get(i);
float splitWidth = getWidth(split);
boolean enoughRoom = state.column() + splitWidth <= maxWidth;
state = computeSplit(commentsHelper, maxWidth, split, state.withMustBreak(false), explorationNode);
if (!enoughRoom) {
state = state.withMustBreak(true);
}
}
return state;
}
private static SplitsBreaks splitByBreaks(List docs) {
ImmutableSplitsBreaks.Builder builder = ImmutableSplitsBreaks.builder();
ImmutableList.Builder currentSplit = ImmutableList.builder();
for (Doc doc : docs) {
if (doc instanceof Break) {
builder.addSplits(currentSplit.build());
currentSplit = ImmutableList.builder();
builder.addBreaks((Break) doc);
} else {
currentSplit.add(doc);
}
}
builder.addSplits(currentSplit.build());
return builder.build();
}
/** Compute breaks for a {@link Level} that spans multiple lines. */
private State computeBroken(
CommentsHelper commentsHelper, int maxWidth, State state, Obs.ExplorationNode explorationNode) {
SplitsBreaks splitsBreaks = memoizedSplitsBreaks.get();
if (!splitsBreaks.breaks().isEmpty()) {
state = state.withBrokenLevel();
}
ImmutableList splitDocs = splitsBreaks.splits().get(0);
state = computeBreakAndSplit(
commentsHelper, maxWidth, state, /* optBreakDoc= */ Optional.empty(), splitDocs, explorationNode);
// Handle following breaks and split.
for (int i = 0; i < splitsBreaks.breaks().size(); i++) {
state = computeBreakAndSplit(
commentsHelper,
maxWidth,
state,
Optional.of(splitsBreaks.breaks().get(i)),
splitsBreaks.splits().get(i + 1),
explorationNode);
}
return state;
}
/** Lay out a Break-separated group of Docs in the current Level. */
private State computeBreakAndSplit(
CommentsHelper commentsHelper,
int maxWidth,
State state,
Optional optBreakDoc,
List split,
Obs.ExplorationNode explorationNode) {
float breakWidth = optBreakDoc.isPresent() ? optBreakDoc.get().getWidth() : 0.0F;
float splitWidth = getWidth(split);
boolean shouldBreak = (optBreakDoc.isPresent() && optBreakDoc.get().fillMode() == FillMode.UNIFIED)
|| state.mustBreak()
|| Double.isInfinite(breakWidth)
|| !tryToFitOnOneLine(maxWidth, state.withColumn(state.column() + (int) breakWidth), split)
.isPresent();
if (optBreakDoc.isPresent()) {
state = optBreakDoc.get().computeBreaks(state, shouldBreak);
}
boolean enoughRoom = state.column() + splitWidth <= maxWidth;
state = computeSplit(commentsHelper, maxWidth, split, state.withMustBreak(false), explorationNode);
if (!enoughRoom) {
state = state.withMustBreak(true); // Break after, too.
}
return state;
}
private static State computeSplit(
CommentsHelper commentsHelper,
int maxWidth,
List docs,
State state,
Obs.ExplorationNode explorationNode) {
for (Doc doc : docs) {
state = doc.computeBreaks(commentsHelper, maxWidth, state, explorationNode);
}
return state;
}
@Override
public void write(State state, Output output) {
if (state.isOneLine(this)) {
output.append(state, getFlat(), range()); // This is defined because width is finite.
} else {
writeFilled(state, output);
}
}
private void writeFilled(State state, Output output) {
SplitsBreaks splitsBreaks = memoizedSplitsBreaks.get();
// Handle first split.
for (Doc doc : splitsBreaks.splits().get(0)) {
doc.write(state, output);
}
// Handle following breaks and split.
for (int i = 0; i < splitsBreaks.breaks().size(); i++) {
splitsBreaks.breaks().get(i).write(state, output);
for (Doc doc : splitsBreaks.splits().get(i + 1)) {
doc.write(state, output);
}
}
}
public Indent getPlusIndent() {
return openOp.plusIndent();
}
BreakBehaviour getBreakBehaviour() {
return openOp.breakBehaviour();
}
List getDocs() {
return docs;
}
LastLevelBreakability getBreakabilityIfLastLevel() {
return openOp.breakabilityIfLastLevel();
}
public PartialInlineability partialInlineability() {
return openOp.partialInlineability();
}
public Optional getDebugName() {
return openOp.debugName();
}
public OpenOp getOpenOp() {
return openOp;
}
/**
* An optional, more restrictive column limit for inner breaks that are marked as {@link Break#hasColumnLimit()}. If
* the level is to be considered one-lineable, the last such break must not start at a column higher than this.
*/
public OptionalInt getColumnLimitBeforeLastBreak() {
return openOp.columnLimitBeforeLastBreak();
}
/** An indented representation of this level and all nested levels inside it. */
public String representation(State state) {
return new LevelDelimitedFlatValueDocVisitor(state).visit(this);
}
/**
* Get the width of a sequence of {@link Doc}s.
*
* @param docs the {@link Doc}s
* @return the width, or {@code Float.POSITIVE_INFINITY} if any {@link Doc} must be broken
*/
static float getWidth(Iterable docs) {
float width = 0.0F;
for (Doc doc : docs) {
width += doc.getWidth();
}
return width;
}
private static Range union(Range x, Range y) {
return x.isEmpty() ? y : y.isEmpty() ? x : x.span(y).canonical(INTEGERS);
}
@Override
public String toString() {
return MoreObjects.toStringHelper(this)
.add("debugName", getDebugName())
.add("plusIndent", getPlusIndent())
.add("breakBehaviour", getBreakBehaviour())
.add("breakabilityIfLastLevel", getBreakabilityIfLastLevel())
.toString();
}
@Target(ElementType.TYPE)
@Retention(RetentionPolicy.SOURCE)
@Value.Style(overshadowImplementation = true)
@interface SplitsBreaksStyle {}
@SplitsBreaksStyle
@Value.Immutable
interface SplitsBreaks {
/** Groups of {@link Doc}s that are children of the current {@link Level}, separated by {@link Break}s. */
ImmutableList> splits();
/** {@link Break}s between {@link Doc}s in the current {@link Level}. */
ImmutableList breaks();
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy