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

net.morimekta.console.terminal.Progress Maven / Gradle / Ivy

package net.morimekta.console.terminal;

import com.google.common.annotations.VisibleForTesting;
import net.morimekta.console.chr.Char;
import net.morimekta.console.chr.Color;
import net.morimekta.console.chr.Control;
import net.morimekta.console.chr.Unicode;
import net.morimekta.util.Strings;

import javax.annotation.Nonnull;
import javax.annotation.Nullable;
import java.time.Clock;
import java.time.Duration;
import java.time.temporal.ChronoUnit;
import java.util.function.IntSupplier;

import static java.lang.Math.max;

/**
 * Show progress on a single task in how many percent (with spinner and
 * progress-bar). Spinner type is configurable. This is the single-thread
 * progress where everything is handled in the same thread as calls
 * {@link #accept(long)}. This class is not thread safe.
 */
public class Progress implements ProgressTask {
    /**
     * Which spinner to show. Some may require extended unicode font to
     * be used in the console without just showing '?'.
     */
    public enum Spinner {
        /**
         * Simple ASCII spinner using '|', '/', '-', '\'. This variant will
         * work in any terminal.
         */
        ASCII(new Unicode('#'),
              new Unicode('-'),
              new Unicode('v'),
              new Unicode[] {
                      new Unicode('|'),
                      new Unicode('/'),
                      new Unicode('-'),
                      new Unicode('\\')
              }),

        /**
         * Using a block char that bounces up and down to show progress.
         * Not exactly spinning, but does the job. Using unicode
         * chars 0x2581 -> 0x2588;
         * 

* '▁', '▂', '▃', '▄', '▅', '▆', '▇', '█' */ BLOCKS(new Unicode('▓'), new Unicode('⋅'), new Unicode('✓'), new Unicode[] { new Unicode('▁'), // 1/8 block new Unicode('▂'), // 2/8 block new Unicode('▃'), // ... new Unicode('▄'), // new Unicode('▅'), // new Unicode('▆'), // ... new Unicode('▇'), // 7/8 block new Unicode('█'), // 8/8 (full) block new Unicode('▇'), // 7/8 block new Unicode('▆'), // ... new Unicode('▅'), // new Unicode('▄'), // new Unicode('▂'), // ... new Unicode('▁'), // 2/8 block }), /** * A spinning arrow. Using chars in range 0x2b60 -> 0x2b69: *

* '⭢', '⭨', '⭣', '⭩', '⭠', '⭦', '⭡', '⭧' */ ARROWS(new Unicode('⬛'), new Unicode('⋅'), new Unicode('✓'), new Unicode[] { new Unicode('⭢'), new Unicode('⭨'), new Unicode('⭣'), new Unicode('⭩'), new Unicode('⭠'), new Unicode('⭦'), new Unicode('⭡'), new Unicode('⭧'), }), /** * Use Unicode clock symbols, 0x1f550 -> 0x1f55b: *

* '🕐', '🕑', '🕒', '🕓', '🕔', '🕕', '🕖', '🕗', '🕘', '🕙', '🕚', '🕛' */ CLOCK(new Unicode('⬛'), new Unicode('⋅'), new Unicode('✓'), new Unicode[] { new Unicode(0x1f550), // 1 o'clock new Unicode(0x1f551), // ... new Unicode(0x1f552), new Unicode(0x1f553), new Unicode(0x1f554), new Unicode(0x1f555), new Unicode(0x1f556), new Unicode(0x1f557), new Unicode(0x1f558), new Unicode(0x1f559), new Unicode(0x1f55a), // ... new Unicode(0x1f55b) // 12 o'clock }), ; Char done; Char remain; Char complete; Char[] spinner; Spinner(Char done, Char remain, Char complete, Char[] spinner) { this.done = done; this.spinner = spinner; this.remain = remain; this.complete = complete; } } private final Terminal terminal; private final Spinner spinner; private final long total; private final long start; private final Clock clock; private final String title; private final LinePrinter updater; private final IntSupplier terminalWidthSupplier; private int spinner_pos; private double fraction; private int last_pct; private int last_pts; private long last_update; private long spinner_update; private long expected_done_ts; /** * Create a progress bar using the given terminal. * * @param terminal The terminal to use. * @param spinner The spinner to use. * @param title The title of the progress. * @param total The total progress value. */ public Progress(@Nonnull Terminal terminal, @Nonnull Spinner spinner, @Nonnull String title, long total) { this(terminal, null, () -> terminal.getTTY().getTerminalSize().cols, Clock.systemUTC(), spinner, title, total); } /** * Create a progress bar using the line printer and width supplier. * * @param updater The line printer used to update visible progress. * @param widthSupplier The width supplier to get terminal width from. * @param spinner The spinner to use. * @param title The title of the progress. * @param total The total progress value. */ public Progress(@Nonnull LinePrinter updater, @Nonnull IntSupplier widthSupplier, @Nonnull Spinner spinner, @Nonnull String title, long total) { this(null, updater, widthSupplier, Clock.systemUTC(), spinner, title, total); } /** * Update the progress to reflect the current progress value. * * @param current The new current progress value. */ @Override public void accept(long current) { if (isDone()) return; long now = clock.millis(); if (current > total) current = total; int pts_w = terminalWidthSupplier.getAsInt() - 23 - title.length(); fraction = ((double) current) / ((double) total); int pct = (int) (fraction * 100); int pts = (int) (fraction * pts_w); if (current < total) { if (now - last_update < 73 && pct == last_pct && pts == last_pts) { return; } int remaining_pts = pts_w - pts; long duration_ms = now - start; Duration remaining = null; // Progress has actually gone forward, recalculate total time. if (duration_ms > 3000) { long remaining_ms; if (expected_done_ts == 0L || pct > last_pct) { long assumed_total = (long) (((double) duration_ms) / fraction); remaining_ms = max(0L, assumed_total - duration_ms); expected_done_ts = now + remaining_ms; } else { remaining_ms = max(0L, expected_done_ts - now); } remaining = Duration.of(remaining_ms, ChronoUnit.MILLIS); } if (now >= (spinner_update + 100)) { spinner_pos = (spinner_pos + 1) % spinner.spinner.length; spinner_update = now; } if (pts < pts_w) { updater.formatln("%s: [%s%s%s%s%s] %3d%% %s%s%s%s", title, Color.GREEN, Strings.times(spinner.done.toString(), pts), Color.YELLOW, Strings.times(spinner.remain.toString(), remaining_pts), Color.CLEAR, pct, new Color(Color.YELLOW, Color.BOLD), spinner.spinner[spinner_pos], Color.CLEAR, remaining == null ? "" : " + " + format(remaining)); } else { updater.formatln("%s: [%s%s%s] 100%% %s%s%s%s", title, Color.GREEN, Strings.times(spinner.done.toString(), pts), Color.CLEAR, new Color(Color.YELLOW, Color.BOLD), spinner.spinner[spinner_pos], Color.CLEAR, remaining == null ? "" : " + " + format(remaining)); } last_pct = pct; last_pts = pts; last_update = now; } else { updater.formatln("%s: [%s%s%s] 100%% %s%s%s @ %s", title, Color.GREEN, Strings.times(spinner.done.toString(), pts_w), Color.CLEAR, new Color(Color.GREEN, Color.BOLD), spinner.complete, Color.CLEAR, format(Duration.of(now - start, ChronoUnit.MILLIS))); last_update = Long.MAX_VALUE; last_pct = 100; } } @Override public void close() { long now = clock.millis(); if (now >= last_update) { int pts_w = terminalWidthSupplier.getAsInt() - 23 - title.length(); int pts = max(0, (int) (fraction * pts_w)); int remaining_pts = max(0, pts_w - pts); updater.formatln("%s: [%s%s%s%s%s] %3d%% %sAborted%s", title, Color.GREEN, Strings.times(spinner.done.toString(), pts), Color.YELLOW, Strings.times(spinner.remain.toString(), remaining_pts), Color.CLEAR, last_pct, new Color(Color.RED, Color.BOLD), Color.CLEAR); last_update = Long.MAX_VALUE; } } @Override public boolean isDone() { return last_update > clock.millis(); } /** * Create a progress updater. Note that either terminal or the * updater param must be set. * * @param terminal The terminal to print to. * @param updater The updater to write to. * @param widthSupplier The width supplier to get terminal width from. * @param clock The clock to use for timing. * @param spinner The spinner type. * @param title What progresses. * @param total The total value to be 'progressed'. */ @VisibleForTesting protected Progress(@Nullable Terminal terminal, @Nullable LinePrinter updater, @Nonnull IntSupplier widthSupplier, @Nonnull Clock clock, @Nonnull Spinner spinner, @Nonnull String title, long total) { this.terminal = terminal; this.terminalWidthSupplier = widthSupplier; this.updater = updater != null ? updater : this::println; this.spinner = spinner; this.title = title; this.total = total; this.start = clock.millis(); this.clock = clock; this.last_pct = -1; this.last_pts = -1; this.spinner_pos = 0; this.spinner_update = start; if (terminal != null) { terminal.finish(); } accept(0); } private void println(String line) { terminal.print("\r" + Control.CURSOR_ERASE + line); } static String format(Duration duration) { long h = duration.toHours(); long m = duration.minusHours(h).toMinutes(); if (h > 0) { return String.format("%2d:%02d H", h, m); } long s = duration.minusHours(h).minusMinutes(m).getSeconds(); if (m > 0) { return String.format("%2d:%02d min", m, s); } long ms = duration.minusHours(h).minusMinutes(m).minusSeconds(s).toMillis(); return String.format("%2d.%1d s", s, ms / 100); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy