blob: fa13fd6453924e5c7613bd636aace3b0c8bc504b [file] [log] [blame]
// Copyright 2016 The Bazel Authors. 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.google.devtools.build.lib.util.io;
import java.io.IOException;
/**
* Wrap an {@link AnsiTerminalWriter} into one that breaks lines to use
* at most a first given number of columns of the terminal. In this way,
* all line breaks are predictable, even if we only have a lower bound
* on the number of columns of the underlying terminal. To simplify copy
* and paste of the terminal output, a continuation character is written
* into the last usable column when we break a line. Additionally, newline
* characters are translated into calls to the {@link AnsiTerminalWriter#newline()}
* method.
*/
public class LineWrappingAnsiTerminalWriter implements AnsiTerminalWriter {
private final AnsiTerminalWriter terminalWriter;
private final int width;
private final char continuationCharacter;
private int position;
public LineWrappingAnsiTerminalWriter(
AnsiTerminalWriter terminalWriter, int width, char continuationCharacter) {
this.terminalWriter = terminalWriter;
this.width = width;
this.continuationCharacter = continuationCharacter;
this.position = 0;
}
public LineWrappingAnsiTerminalWriter(AnsiTerminalWriter terminalWriter, int width) {
this(terminalWriter, width, '\\');
}
private void appendChar(char c) throws IOException {
if (c == '\n') {
terminalWriter.newline();
position = 0;
} else if (position + 1 < width) {
terminalWriter.append(Character.toString(c));
position++;
} else {
// The last usable character of the line was already been written,
// hence we have to start a continuation before writing the symbol.
terminalWriter.append(Character.toString(continuationCharacter));
terminalWriter.newline();
terminalWriter.append(Character.toString(c));
position = 1;
}
}
@Override
public AnsiTerminalWriter append(String text) throws IOException {
for (int i = 0; i < text.length(); i++) {
appendChar(text.charAt(i));
}
return this;
}
@Override
public AnsiTerminalWriter newline() throws IOException {
terminalWriter.newline();
position = 0;
return this;
}
@Override
public AnsiTerminalWriter okStatus() throws IOException {
terminalWriter.okStatus();
return this;
}
@Override
public AnsiTerminalWriter failStatus() throws IOException {
terminalWriter.failStatus();
return this;
}
@Override
public AnsiTerminalWriter normal() throws IOException {
terminalWriter.normal();
return this;
}
}