removed everything not explicitly zeitlaeufer
This commit is contained in:
		@@ -1,415 +0,0 @@
 | 
			
		||||
import java.io.BufferedReader;
 | 
			
		||||
import java.io.InputStreamReader;
 | 
			
		||||
import java.io.IOException;
 | 
			
		||||
import java.math.BigDecimal;
 | 
			
		||||
import java.math.MathContext;
 | 
			
		||||
import java.nio.charset.StandardCharsets;
 | 
			
		||||
import java.text.DecimalFormat;
 | 
			
		||||
import java.text.DecimalFormatSymbols;
 | 
			
		||||
import java.text.ParseException;
 | 
			
		||||
import java.time.Month;
 | 
			
		||||
import java.time.YearMonth;
 | 
			
		||||
import java.util.Locale;
 | 
			
		||||
 | 
			
		||||
class Darlehenberechner {
 | 
			
		||||
 | 
			
		||||
   private static final class Konfiguration {
 | 
			
		||||
 | 
			
		||||
      private BigDecimal darlehenswert;
 | 
			
		||||
      private BigDecimal zinssatzReal;
 | 
			
		||||
      private BigDecimal monatlicheRate;
 | 
			
		||||
      private Integer laufzeitMonate;
 | 
			
		||||
      private BigDecimal restschuld;
 | 
			
		||||
      private Integer tilgungsfreieZeit;
 | 
			
		||||
      private YearMonth anfangsmonat;
 | 
			
		||||
      private BigDecimal sondertilgungReal;
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
      public BigDecimal getDarlehenswert() {
 | 
			
		||||
         return darlehenswert;
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
      public Konfiguration setDarlehenswert(BigDecimal darlehenswert) {
 | 
			
		||||
         this.darlehenswert = darlehenswert;
 | 
			
		||||
         return this;
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
      public BigDecimal getZinssatz() {
 | 
			
		||||
         return zinssatzReal;
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
      public Konfiguration setZinssatzProzent(BigDecimal zinssatzProzent) {
 | 
			
		||||
         this.zinssatzReal = zinssatzProzent.divide(EINHUNDERT, MathContext.DECIMAL128);
 | 
			
		||||
         return this;
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
      public BigDecimal getMonatlicheRate() {
 | 
			
		||||
         return monatlicheRate;
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
      public Konfiguration setMonatlicheRate(BigDecimal monatlicheRate) {
 | 
			
		||||
         this.monatlicheRate = monatlicheRate;
 | 
			
		||||
         return this;
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
      public Integer getLaufzeitMonate() {
 | 
			
		||||
         return laufzeitMonate;
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
      public Konfiguration setLaufzeitMonate(Integer laufzeitMonate) {
 | 
			
		||||
         this.laufzeitMonate = laufzeitMonate;
 | 
			
		||||
         return this;
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
      public Konfiguration setLaufzeitJahre(Integer jahre) {
 | 
			
		||||
         this.laufzeitMonate = (jahre * 12);
 | 
			
		||||
         return this;
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
      public Konfiguration setLaufzeit(Integer jahre, Integer monate) {
 | 
			
		||||
         this.laufzeitMonate = (jahre * 12) + monate;
 | 
			
		||||
         return this;
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
      public BigDecimal getRestschuld() {
 | 
			
		||||
         return restschuld;
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
      public Konfiguration setRestschuld(BigDecimal restschuld) {
 | 
			
		||||
         this.restschuld = restschuld;
 | 
			
		||||
         return this;
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
      public Integer getTilgungsfreieZeit() {
 | 
			
		||||
         return tilgungsfreieZeit;
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
      public Konfiguration setTilgungsfreieZeit(Integer tilgungsfreieZeit) {
 | 
			
		||||
         this.tilgungsfreieZeit = tilgungsfreieZeit;
 | 
			
		||||
         return this;
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
      public YearMonth getAnfangsmonat() {
 | 
			
		||||
         return anfangsmonat;
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
      public Konfiguration setAnfangsmonat(YearMonth anfangsmonat) {
 | 
			
		||||
         this.anfangsmonat = anfangsmonat;
 | 
			
		||||
         return this;
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
      public BigDecimal getSondertilgung() {
 | 
			
		||||
         return sondertilgungReal;
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
      public Konfiguration setSondertilgungProzent(BigDecimal sondertilgungProzent) {
 | 
			
		||||
         this.sondertilgungReal = sondertilgungProzent.divide(EINHUNDERT, MathContext.DECIMAL128);
 | 
			
		||||
         return this;
 | 
			
		||||
      }
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
   private static final DecimalFormat DECIMAL_FORMAT = new DecimalFormat("#,##0.00", new DecimalFormatSymbols(Locale.GERMAN));
 | 
			
		||||
   private static final BigDecimal ZWOELF = BigDecimal.valueOf(12);
 | 
			
		||||
   private static final BigDecimal EINHUNDERT = BigDecimal.valueOf(100);
 | 
			
		||||
 | 
			
		||||
   private final Integer laufzeitMonate;
 | 
			
		||||
   private final BigDecimal restschuld;
 | 
			
		||||
   private final BigDecimal zinssatz;
 | 
			
		||||
   private BigDecimal sondertilgung;
 | 
			
		||||
   private int summeMonate = 0;
 | 
			
		||||
   private BigDecimal aktRestschuld;
 | 
			
		||||
   private BigDecimal aktMonatlicheRate;
 | 
			
		||||
   private BigDecimal aktZinsbetrag;
 | 
			
		||||
   private BigDecimal aktTilgungsbetrag;
 | 
			
		||||
   private Integer aktTilgungsfreieZeit;
 | 
			
		||||
   private YearMonth aktMonat;
 | 
			
		||||
   private BigDecimal jahressummeZinsenKalenderjahr = BigDecimal.ZERO;
 | 
			
		||||
   private BigDecimal jahressummeZinsenKreditjahr = BigDecimal.ZERO;
 | 
			
		||||
   private BigDecimal summeZinsen = BigDecimal.ZERO;
 | 
			
		||||
   private BigDecimal jahressummeTilgungKalenderjahr = BigDecimal.ZERO;
 | 
			
		||||
   private BigDecimal jahressummeTilgungKreditjahr = BigDecimal.ZERO;
 | 
			
		||||
   private BigDecimal summeTilgung = BigDecimal.ZERO;
 | 
			
		||||
   private BigDecimal jahressummeRatenKalenderjahr = BigDecimal.ZERO;
 | 
			
		||||
   private BigDecimal jahressummeRatenKreditjahr = BigDecimal.ZERO;
 | 
			
		||||
   private BigDecimal summeRaten = BigDecimal.ZERO;
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   public static void main(String[] args) throws ParseException, IOException {
 | 
			
		||||
      /*new Darlehenberechner(new Konfiguration()
 | 
			
		||||
         .setDarlehenswert(BigDecimal.valueOf(168_000))
 | 
			
		||||
         .setZinssatzProzent(BigDecimal.valueOf(3.73))
 | 
			
		||||
         .setMonatlicheRate(BigDecimal.valueOf(1_500))
 | 
			
		||||
         .setTilgungsfreieZeit(0)
 | 
			
		||||
         .setAnfangsmonat(YearMonth.of(2024, Month.SEPTEMBER))
 | 
			
		||||
         .setSondertilgungProzent(BigDecimal.valueOf(2.5))
 | 
			
		||||
      ).berechneWerte();*/
 | 
			
		||||
 | 
			
		||||
      /*new Darlehenberechner(new Konfiguration()
 | 
			
		||||
         .setDarlehenswert(BigDecimal.valueOf(168_000))
 | 
			
		||||
         .setZinssatzProzent(BigDecimal.valueOf(3.73))
 | 
			
		||||
         .setMonatlicheRate(BigDecimal.valueOf(1_500))
 | 
			
		||||
         .setTilgungsfreieZeit(0)
 | 
			
		||||
         .setLaufzeitJahre(11)
 | 
			
		||||
         .setAnfangsmonat(YearMonth.of(2024, Month.SEPTEMBER))
 | 
			
		||||
      ).berechneWerte();*/
 | 
			
		||||
 | 
			
		||||
      if (args.length == 0) {
 | 
			
		||||
         askParametersAndRun();
 | 
			
		||||
      } else {
 | 
			
		||||
         parseParametersAndRun(args);
 | 
			
		||||
      }
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   private static void askParametersAndRun() throws ParseException, IOException {
 | 
			
		||||
      DECIMAL_FORMAT.setParseBigDecimal(true);
 | 
			
		||||
 | 
			
		||||
      var konfig = new Konfiguration();
 | 
			
		||||
      var br = new BufferedReader(new InputStreamReader(System.in, StandardCharsets.UTF_8));
 | 
			
		||||
      System.out.print("Darlehenswert: ");
 | 
			
		||||
      konfig.setDarlehenswert((BigDecimal) DECIMAL_FORMAT.parse(br.readLine()));
 | 
			
		||||
      System.out.print("Zinssatz: ");
 | 
			
		||||
      konfig.setZinssatzProzent((BigDecimal) DECIMAL_FORMAT.parse(br.readLine()));
 | 
			
		||||
      System.out.print("Monatliche Rate: ");
 | 
			
		||||
      konfig.setMonatlicheRate((BigDecimal) DECIMAL_FORMAT.parse(br.readLine()));
 | 
			
		||||
      System.out.print("Monat erste Rate(z.B. 2007-12): ");
 | 
			
		||||
      konfig.setAnfangsmonat(YearMonth.parse(br.readLine()));
 | 
			
		||||
      System.out.print("Laufzeit in Jahren(optional Jahre:Monate): ");
 | 
			
		||||
      String in = br.readLine();
 | 
			
		||||
      if (in != null && !in.isBlank()) {
 | 
			
		||||
         String[] split = in.split(":");
 | 
			
		||||
         konfig.setLaufzeit(Integer.parseInt(split[0]), Integer.parseInt(split[1]));
 | 
			
		||||
      } else {
 | 
			
		||||
         System.out.print("Restschuld(optional): ");
 | 
			
		||||
         in = br.readLine();
 | 
			
		||||
         if (in != null && !in.isBlank()) {
 | 
			
		||||
            konfig.setRestschuld((BigDecimal) DECIMAL_FORMAT.parse(in));
 | 
			
		||||
         }
 | 
			
		||||
      }
 | 
			
		||||
      System.out.print("Anzahl tilgungsfreier Monate(optional): ");
 | 
			
		||||
      in = br.readLine();
 | 
			
		||||
      if (in != null && !in.isBlank()) {
 | 
			
		||||
         konfig.setTilgungsfreieZeit(Integer.parseInt(in));
 | 
			
		||||
      }
 | 
			
		||||
      System.out.print("Sondertilgungssatz(optional): ");
 | 
			
		||||
      in = br.readLine();
 | 
			
		||||
      if (in != null && !in.isBlank()) {
 | 
			
		||||
         konfig.setSondertilgungProzent((BigDecimal) DECIMAL_FORMAT.parse(in));
 | 
			
		||||
      }
 | 
			
		||||
      new Darlehenberechner(konfig).berechneWerte();
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   private static void parseParametersAndRun(String[] args) throws ParseException {
 | 
			
		||||
      var konfig = new Konfiguration();
 | 
			
		||||
      int count = 0;
 | 
			
		||||
      DECIMAL_FORMAT.setParseBigDecimal(true);
 | 
			
		||||
      while (count < args.length) {
 | 
			
		||||
         String arg = args[count];
 | 
			
		||||
         if (arg.equals("-hilfe")) {
 | 
			
		||||
            System.out.println("-darlehenswert 1000,00 -zinssatz 3,73 -monatlicheRate 30,00 -anfangsmonat 2024-09"
 | 
			
		||||
               + "[-laufzeitJahre 11] [-aktTilgungsfreieZeit 5]");
 | 
			
		||||
         }
 | 
			
		||||
         if (arg.equals("-darlehenswert")) {
 | 
			
		||||
            count++;
 | 
			
		||||
            konfig.setDarlehenswert((BigDecimal) DECIMAL_FORMAT.parse(args[count]));
 | 
			
		||||
         }
 | 
			
		||||
         if (arg.equals("-zinssatz")) {
 | 
			
		||||
            count++;
 | 
			
		||||
            konfig.setZinssatzProzent((BigDecimal) DECIMAL_FORMAT.parse(args[count]));
 | 
			
		||||
         }
 | 
			
		||||
         if (arg.equals("-sondertilgung")) {
 | 
			
		||||
            count++;
 | 
			
		||||
            konfig.setSondertilgungProzent((BigDecimal) DECIMAL_FORMAT.parse(args[count]));
 | 
			
		||||
         }
 | 
			
		||||
         if (arg.equals("-monatlicheRate")) {
 | 
			
		||||
            count++;
 | 
			
		||||
            konfig.setMonatlicheRate((BigDecimal) DECIMAL_FORMAT.parse(args[count]));
 | 
			
		||||
         }
 | 
			
		||||
         if (arg.equals("-aktTilgungsfreieZeit")) {
 | 
			
		||||
            count++;
 | 
			
		||||
            konfig.setTilgungsfreieZeit(Integer.parseInt(args[count]));
 | 
			
		||||
         }
 | 
			
		||||
         if (arg.equals("-laufzeitJahre")) {
 | 
			
		||||
            count++;
 | 
			
		||||
            konfig.setLaufzeitJahre(Integer.parseInt(args[count]));
 | 
			
		||||
         }
 | 
			
		||||
         if (arg.equals("-anfangsmonat")) {
 | 
			
		||||
            count++;
 | 
			
		||||
            konfig.setAnfangsmonat(YearMonth.parse(args[count]));
 | 
			
		||||
         }
 | 
			
		||||
         count++;
 | 
			
		||||
      }
 | 
			
		||||
      new Darlehenberechner(konfig).berechneWerte();
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   private Darlehenberechner(Konfiguration konfig) {
 | 
			
		||||
      laufzeitMonate = konfig.getLaufzeitMonate();
 | 
			
		||||
      restschuld = konfig.getRestschuld();
 | 
			
		||||
      zinssatz = konfig.getZinssatz();
 | 
			
		||||
      sondertilgung = konfig.getSondertilgung() != null ? konfig.getSondertilgung().multiply(konfig.getDarlehenswert()) : BigDecimal.ZERO;
 | 
			
		||||
      aktRestschuld = konfig.getDarlehenswert();
 | 
			
		||||
      if (restschuld != null) {
 | 
			
		||||
         aktRestschuld = aktRestschuld.subtract(restschuld);
 | 
			
		||||
      }
 | 
			
		||||
      aktMonatlicheRate = konfig.getMonatlicheRate();
 | 
			
		||||
      aktTilgungsfreieZeit = konfig.getTilgungsfreieZeit();
 | 
			
		||||
      aktMonat = konfig.getAnfangsmonat();
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   private void berechneWerte() {
 | 
			
		||||
      druckeUeberschrift();
 | 
			
		||||
      while (laufzeitNichtVorbei()) {
 | 
			
		||||
         erhoeheSummeMonate();
 | 
			
		||||
         zahleSondertilgung();
 | 
			
		||||
         berechneBeitragAufteilung();
 | 
			
		||||
         berechneRestschuld();
 | 
			
		||||
         druckeAktuelleMonatswerte();
 | 
			
		||||
         berechneSummen();
 | 
			
		||||
         druckeJahressumeBedingt();
 | 
			
		||||
         aktualisiereZeitwerte();
 | 
			
		||||
      }
 | 
			
		||||
      druckeFinaleSummen();
 | 
			
		||||
      druckeLaufzeitUndRestschuld();
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   private void druckeUeberschrift() {
 | 
			
		||||
      System.out.println("Monat:   Rate     = Zinsen + Tilgung| Restschuld");
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   private boolean laufzeitNichtVorbei() {
 | 
			
		||||
      return laufzeitMonate != null ? summeMonate < laufzeitMonate : aktRestschuld.signum() > 0;
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   private boolean laufzeitVorbei() {
 | 
			
		||||
      return !laufzeitNichtVorbei();
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   private void erhoeheSummeMonate() {
 | 
			
		||||
      summeMonate++;
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   private void zahleSondertilgung() {
 | 
			
		||||
      if (sondertilgung == null || sondertilgung.signum() == 0) {
 | 
			
		||||
         return;
 | 
			
		||||
      }
 | 
			
		||||
      boolean sondertilgungFaellingErstesJahr = summeMonate < 12 && aktMonat.getMonth() == Month.DECEMBER;
 | 
			
		||||
      if (sondertilgungFaellingErstesJahr || summeMonate > 1 && aktMonat.getMonth() == Month.JANUARY) {
 | 
			
		||||
         aktRestschuld = aktRestschuld.compareTo(sondertilgung) > 0 ? aktRestschuld.subtract(sondertilgung) : aktRestschuld;
 | 
			
		||||
         System.out.println(aktMonat + ": " + DECIMAL_FORMAT.format(sondertilgung) + " = 0,00 + " + DECIMAL_FORMAT.format(sondertilgung) + " | " + DECIMAL_FORMAT.format(getRealeRestschuld()));
 | 
			
		||||
      }
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   private BigDecimal getRealeRestschuld() {
 | 
			
		||||
      return restschuld != null ? aktRestschuld.add(restschuld) : aktRestschuld;
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   private void berechneBeitragAufteilung() {
 | 
			
		||||
      aktZinsbetrag = getRealeRestschuld().multiply(zinssatz)
 | 
			
		||||
         .divide(ZWOELF, MathContext.DECIMAL128);
 | 
			
		||||
      if (aktMonatlicheRate.compareTo(aktRestschuld) > 0) {
 | 
			
		||||
         aktMonatlicheRate = aktRestschuld.add(aktZinsbetrag); // die letzte Rate ist gleich der Restschuld + Zinsen
 | 
			
		||||
      }
 | 
			
		||||
      aktTilgungsbetrag = aktTilgungsfreieZeit != null && aktTilgungsfreieZeit > 0 ? BigDecimal.ZERO : aktMonatlicheRate.subtract(aktZinsbetrag);
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   private void berechneRestschuld() {
 | 
			
		||||
      aktRestschuld = aktRestschuld.subtract(aktTilgungsbetrag);
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   private void druckeAktuelleMonatswerte() {
 | 
			
		||||
      System.out.println(aktMonat + ": " + DECIMAL_FORMAT.format(aktMonatlicheRate) + " = " + DECIMAL_FORMAT.format(aktZinsbetrag)
 | 
			
		||||
            + " + " + DECIMAL_FORMAT.format(aktTilgungsbetrag) + " | " + DECIMAL_FORMAT.format(getRealeRestschuld()));
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   private void berechneSummen() {
 | 
			
		||||
      jahressummeZinsenKalenderjahr = jahressummeZinsenKalenderjahr.add(aktZinsbetrag);
 | 
			
		||||
      jahressummeZinsenKreditjahr = jahressummeZinsenKreditjahr.add(aktZinsbetrag);
 | 
			
		||||
      summeZinsen = summeZinsen.add(aktZinsbetrag);
 | 
			
		||||
      jahressummeTilgungKalenderjahr = jahressummeTilgungKalenderjahr.add(aktTilgungsbetrag);
 | 
			
		||||
      jahressummeTilgungKreditjahr = jahressummeTilgungKreditjahr.add(aktTilgungsbetrag);
 | 
			
		||||
      summeTilgung = summeTilgung.add(aktTilgungsbetrag);
 | 
			
		||||
      jahressummeRatenKalenderjahr = jahressummeRatenKalenderjahr.add(aktMonatlicheRate);
 | 
			
		||||
      jahressummeRatenKreditjahr = jahressummeRatenKreditjahr.add(aktMonatlicheRate);
 | 
			
		||||
      summeRaten = summeRaten.add(aktMonatlicheRate);
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   private void druckeJahressumeBedingt() {
 | 
			
		||||
      boolean kreditjahrVergangen = summeMonate > 1 && summeMonate % 12 == 0 || laufzeitVorbei();
 | 
			
		||||
      boolean kalenderjahrVergangen = aktMonat.getMonth() == Month.DECEMBER;
 | 
			
		||||
      if (kreditjahrVergangen || kalenderjahrVergangen) {
 | 
			
		||||
         if (kreditjahrVergangen) {
 | 
			
		||||
            String desc = "Kreditjahr " + (summeMonate + 11) / 12; // + 11 weil integerdivision und X Jahre plus 1 Monat soll X + 1 tes Kreditjahr ergeben
 | 
			
		||||
            druckeJahressumme(desc, jahressummeRatenKreditjahr, jahressummeZinsenKreditjahr, jahressummeTilgungKreditjahr);
 | 
			
		||||
            jahressummeRatenKreditjahr = BigDecimal.ZERO;
 | 
			
		||||
            jahressummeZinsenKreditjahr = BigDecimal.ZERO;
 | 
			
		||||
            jahressummeTilgungKreditjahr = BigDecimal.ZERO;
 | 
			
		||||
         }
 | 
			
		||||
         if (kalenderjahrVergangen) {
 | 
			
		||||
            String desc = "Kalenderjahr " + aktMonat.getYear();
 | 
			
		||||
            druckeJahressumme(desc, jahressummeRatenKalenderjahr, jahressummeZinsenKalenderjahr, jahressummeTilgungKalenderjahr);
 | 
			
		||||
            jahressummeZinsenKalenderjahr = BigDecimal.ZERO;
 | 
			
		||||
            jahressummeTilgungKalenderjahr = BigDecimal.ZERO;
 | 
			
		||||
            jahressummeRatenKalenderjahr = BigDecimal.ZERO;
 | 
			
		||||
         }
 | 
			
		||||
      }
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   private void druckeJahressumme(String desc, BigDecimal jahressummeRaten, BigDecimal jahressummeZinsen, BigDecimal jahressummeTilgung) {
 | 
			
		||||
      System.out.println("Summe " + desc + ":\n" + DECIMAL_FORMAT.format(jahressummeRaten) + " = " + DECIMAL_FORMAT.format(jahressummeZinsen)
 | 
			
		||||
            + " + " + DECIMAL_FORMAT.format(jahressummeTilgung));
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   private void aktualisiereZeitwerte() {
 | 
			
		||||
      aktMonat = aktMonat.plusMonths(1);
 | 
			
		||||
      if (aktTilgungsfreieZeit != null) {
 | 
			
		||||
         aktTilgungsfreieZeit--;
 | 
			
		||||
      }
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   private void druckeFinaleSummen() {
 | 
			
		||||
      System.out.println("Summe:\n" + DECIMAL_FORMAT.format(summeRaten) + " = " + DECIMAL_FORMAT.format(summeZinsen)
 | 
			
		||||
         + " + " + DECIMAL_FORMAT.format(summeTilgung));
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   private void druckeLaufzeitUndRestschuld() {
 | 
			
		||||
      int laufzeitJahreFinal = summeMonate / 12;
 | 
			
		||||
      int teillaufzeitMonateFinal = summeMonate % 12;
 | 
			
		||||
      System.out.println("Laufzeit: " + laufzeitJahreFinal + " Jahre " + teillaufzeitMonateFinal + " Monate");
 | 
			
		||||
      System.out.println("Restschuld: " + DECIMAL_FORMAT.format(getRealeRestschuld()));
 | 
			
		||||
   }
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										165
									
								
								DrinkingBar.java
									
									
									
									
									
								
							
							
						
						
									
										165
									
								
								DrinkingBar.java
									
									
									
									
									
								
							@@ -1,165 +0,0 @@
 | 
			
		||||
import java.io.BufferedReader;
 | 
			
		||||
import java.io.InputStreamReader;
 | 
			
		||||
import java.io.IOException;
 | 
			
		||||
import java.math.BigDecimal;
 | 
			
		||||
import java.math.MathContext;
 | 
			
		||||
import java.math.RoundingMode;
 | 
			
		||||
import java.nio.charset.StandardCharsets;
 | 
			
		||||
import java.text.DecimalFormat;
 | 
			
		||||
import java.time.LocalTime;
 | 
			
		||||
import java.time.format.DateTimeFormatter;
 | 
			
		||||
import java.time.temporal.ChronoUnit;
 | 
			
		||||
import java.util.concurrent.TimeUnit;
 | 
			
		||||
 | 
			
		||||
public class DrinkingBar {
 | 
			
		||||
 | 
			
		||||
   private static final DateTimeFormatter TIME_FORMATTER = DateTimeFormatter.ofPattern("HH:mm");
 | 
			
		||||
   private static final int MINS_PER_HOUR = 60;
 | 
			
		||||
   private static final int MINS_PER_HALF_HOUR = MINS_PER_HOUR / 2;
 | 
			
		||||
   private static final int MINUTES_BEFORE_PAUSE = 4 * MINS_PER_HOUR + MINS_PER_HALF_HOUR;
 | 
			
		||||
   private static final int MINUTES_WITH_PAUSE = 6 * MINS_PER_HOUR;
 | 
			
		||||
   private static final int DEFAULT_TOTAL_TIME = 8 * MINS_PER_HOUR + MINS_PER_HALF_HOUR;
 | 
			
		||||
   private static final BigDecimal DEFAULT_TOTAL_TIME_BD = BigDecimal.valueOf(DEFAULT_TOTAL_TIME);
 | 
			
		||||
   private static final BigDecimal DEFAULT_TOTAL_LITRES = BigDecimal.valueOf(2.0);
 | 
			
		||||
   private static final BigDecimal QUARTER_LITRE = BigDecimal.valueOf(0.25);
 | 
			
		||||
   private static final DecimalFormat LITRE_FORMAT = new DecimalFormat("0.00");
 | 
			
		||||
   private static final DecimalFormat PERCENTAGE_FORMAT = new DecimalFormat("00.00");
 | 
			
		||||
   private static final BigDecimal MINS_PER_HOUR_BD = BigDecimal.valueOf(MINS_PER_HOUR);
 | 
			
		||||
   private static final MathContext MC_INTEGER = new MathContext(1, RoundingMode.HALF_EVEN);
 | 
			
		||||
 | 
			
		||||
   private final LocalTime startTime;
 | 
			
		||||
   private LocalTime endTime;
 | 
			
		||||
   private long totalMinutes;
 | 
			
		||||
   private BigDecimal totalMinutesBD;
 | 
			
		||||
   private BigDecimal totalLitres;
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   private DrinkingBar(LocalTime startTime) {
 | 
			
		||||
      this.startTime = startTime;
 | 
			
		||||
      this.totalMinutes = DEFAULT_TOTAL_TIME;
 | 
			
		||||
      this.totalMinutesBD = BigDecimal.valueOf(totalMinutes);
 | 
			
		||||
      this.endTime = startTime.plusMinutes(totalMinutes);
 | 
			
		||||
      this.totalLitres = DEFAULT_TOTAL_LITRES;
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   public static void main(String[] args) throws IOException {
 | 
			
		||||
      var br = new BufferedReader(new InputStreamReader(System.in, StandardCharsets.UTF_8));
 | 
			
		||||
      print("Ankunftszeit: ");
 | 
			
		||||
      var startTime = LocalTime.parse(br.readLine(), TIME_FORMATTER).truncatedTo(ChronoUnit.MINUTES);
 | 
			
		||||
      var db = new DrinkingBar(startTime);
 | 
			
		||||
      db.showLoadingBar();
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   private static void println(Object o) {
 | 
			
		||||
      System.out.println(o);
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   private static void print(Object o) {
 | 
			
		||||
      System.out.print(o);
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   protected long getPassedMinutes() {
 | 
			
		||||
      return startTime.until(LocalTime.now().truncatedTo(ChronoUnit.MINUTES), ChronoUnit.MINUTES);
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   private void setEndTime(LocalTime endTime) {
 | 
			
		||||
      this.endTime = endTime;
 | 
			
		||||
      this.totalMinutes = startTime.until(endTime, ChronoUnit.MINUTES);
 | 
			
		||||
      this.totalMinutesBD = BigDecimal.valueOf(totalMinutes);
 | 
			
		||||
      extraInitEndTimeTotalMinutes();
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   protected void extraInitEndTimeTotalMinutes() {
 | 
			
		||||
      // correct necessary litres to drink based on the end time.
 | 
			
		||||
      // lower the volume in quarter litre steps
 | 
			
		||||
      BigDecimal calcTotalLitres = DEFAULT_TOTAL_LITRES;
 | 
			
		||||
      BigDecimal totalLitresFromMinutes = DEFAULT_TOTAL_LITRES
 | 
			
		||||
         .multiply(totalMinutesBD) // reverse dreisatz
 | 
			
		||||
         .divide(DEFAULT_TOTAL_TIME_BD, MathContext.DECIMAL64);
 | 
			
		||||
      do {
 | 
			
		||||
         calcTotalLitres = calcTotalLitres.subtract(QUARTER_LITRE);
 | 
			
		||||
      } while (calcTotalLitres.compareTo(totalLitresFromMinutes) >= 0);
 | 
			
		||||
      // add quarter since we always did a step "too many", due to the do ... while loop
 | 
			
		||||
      this.totalLitres = calcTotalLitres.add(QUARTER_LITRE);
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   private void showLoadingBar() {
 | 
			
		||||
      long passedMinutes = getPassedMinutes();
 | 
			
		||||
      // long passedMinutes = 0; // DEBUG
 | 
			
		||||
      if (passedMinutes > totalMinutes) {
 | 
			
		||||
         passedMinutes = totalMinutes;
 | 
			
		||||
      } else if (passedMinutes < 0) {
 | 
			
		||||
         var now = LocalTime.now().truncatedTo(ChronoUnit.SECONDS);
 | 
			
		||||
         println("!ACHTUNG! Startzeit \"" + startTime + ":00\" liegt in der Zukunft von jetzt an (" + now + ") gesehen.");
 | 
			
		||||
      }
 | 
			
		||||
      println(minutesToTimeString(totalMinutesBD) + " gesamt; Endzeit: " + TIME_FORMATTER.format(endTime));
 | 
			
		||||
      while (passedMinutes < totalMinutes) {
 | 
			
		||||
         print(fillLoadingBar(passedMinutes, true));
 | 
			
		||||
         waitUntilNextMinute();
 | 
			
		||||
         passedMinutes++;
 | 
			
		||||
      }
 | 
			
		||||
      println(fillLoadingBar(passedMinutes, false));
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   private String fillLoadingBar(long passedMinutes, boolean progressive) {
 | 
			
		||||
      long effectivePassedMinutes = passedMinutes < 0 ? 0 : passedMinutes;
 | 
			
		||||
      if (totalMinutes > MINUTES_WITH_PAUSE && passedMinutes > MINUTES_BEFORE_PAUSE && passedMinutes <= MINUTES_WITH_PAUSE) {
 | 
			
		||||
         effectivePassedMinutes = MINUTES_BEFORE_PAUSE;
 | 
			
		||||
      }
 | 
			
		||||
      var effectivePassedMinutesBD = BigDecimal.valueOf(effectivePassedMinutes);
 | 
			
		||||
      BigDecimal currentLitres = totalLitres
 | 
			
		||||
         .multiply(effectivePassedMinutesBD) // reverse dreisatz
 | 
			
		||||
         .divide(totalMinutesBD, MathContext.DECIMAL64)
 | 
			
		||||
         .add(QUARTER_LITRE);
 | 
			
		||||
      BigDecimal printedLitres = currentLitres.subtract(currentLitres.remainder(QUARTER_LITRE, MathContext.DECIMAL64));
 | 
			
		||||
      /* BigDecimal currentProgressToNextStep = ONE_HUNDRED_PERCENT
 | 
			
		||||
         .multiply(currentLitres.subtract(printedLitres)) // reverse dreisatz
 | 
			
		||||
         .divide(QUARTER_LITRE, MathContext.DECIMAL64); */
 | 
			
		||||
      BigDecimal minutesToNextStep = getMinutesToNextStep(currentLitres);
 | 
			
		||||
      String progressivePart = progressive ? "\r" : "";
 | 
			
		||||
      return progressivePart + "Aktuelles Volumen: " + LITRE_FORMAT.format(printedLitres) + "L - "
 | 
			
		||||
         // + PERCENTAGE_FORMAT.format(currentProgressToNextStep) + "% - "
 | 
			
		||||
         + minutesToTimeString(minutesToNextStep);
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   private BigDecimal getMinutesToNextStep(BigDecimal currentLitres) {
 | 
			
		||||
      // berechne Liter benötigt bis zum nächsten 0.25er Schritt
 | 
			
		||||
      BigDecimal litresToNextStep = QUARTER_LITRE.subtract(currentLitres.remainder(QUARTER_LITRE));
 | 
			
		||||
      // berechne Minuten benötigt für 1 Liter
 | 
			
		||||
      BigDecimal minutesPerLitre = totalMinutesBD.divide(totalLitres, MathContext.DECIMAL64);
 | 
			
		||||
      // berechne Minuten benötigt bis zum nächsten 0.25er Schritt
 | 
			
		||||
      // runde auf ganze Zahl, da wir nur ganze Minuten anzeigen und damit 1.999 = 2 Minuten sind
 | 
			
		||||
      return minutesPerLitre.multiply(litresToNextStep).setScale(0, RoundingMode.HALF_EVEN);
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   private String minutesToTimeString(BigDecimal minutes) {
 | 
			
		||||
      BigDecimal[] hoursAndMinutes = minutes.divideAndRemainder(MINS_PER_HOUR_BD, MC_INTEGER);
 | 
			
		||||
      return LocalTime.of(hoursAndMinutes[0].intValue(), hoursAndMinutes[1].intValue()).format(TIME_FORMATTER);
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   private void waitUntilNextMinute() {
 | 
			
		||||
      try {
 | 
			
		||||
         var now = LocalTime.now();
 | 
			
		||||
         var oneMinuteLater = now.plusMinutes(1).truncatedTo(ChronoUnit.MINUTES);
 | 
			
		||||
         /* We wait whole seconds to not make it overly complicated.
 | 
			
		||||
            That results in cut milliseconds: if we would have to wait 1 second and 526 milliseconds, we wait only 1 second.
 | 
			
		||||
            So, adjust for ignored milliseconds, add +1 second as it is better to switch between 00 and 01 as between 59 and 00 */
 | 
			
		||||
         TimeUnit.SECONDS.sleep(now.until(oneMinuteLater, ChronoUnit.SECONDS) + 1);
 | 
			
		||||
         // TimeUnit.MILLISECONDS.sleep(100L); // DEBUG
 | 
			
		||||
      } catch (InterruptedException ie) {
 | 
			
		||||
         Thread.currentThread().interrupt();
 | 
			
		||||
         throw new RuntimeException(ie);
 | 
			
		||||
      }
 | 
			
		||||
   }
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										268
									
								
								FabianUtil.java
									
									
									
									
									
								
							
							
						
						
									
										268
									
								
								FabianUtil.java
									
									
									
									
									
								
							@@ -1,268 +0,0 @@
 | 
			
		||||
package de.vorsorge.theo.util;
 | 
			
		||||
 | 
			
		||||
import java.io.IOException;
 | 
			
		||||
import java.math.BigDecimal;
 | 
			
		||||
import java.math.MathContext;
 | 
			
		||||
import java.nio.charset.StandardCharsets;
 | 
			
		||||
import java.nio.file.Files;
 | 
			
		||||
import java.nio.file.Paths;
 | 
			
		||||
import java.text.DecimalFormat;
 | 
			
		||||
import java.time.Duration;
 | 
			
		||||
import java.time.ZonedDateTime;
 | 
			
		||||
import java.time.temporal.ChronoUnit;
 | 
			
		||||
import java.util.concurrent.TimeUnit;
 | 
			
		||||
 | 
			
		||||
public class FabianUtil {
 | 
			
		||||
 | 
			
		||||
   private static final String STD_DIR_TEMP = "/FIXME/temp/";
 | 
			
		||||
   public static final String ENDING_TXT = "txt";
 | 
			
		||||
   public static final String ENDING_JSON = "json";
 | 
			
		||||
   public static final String ENDING_HTML = "html";
 | 
			
		||||
   public static final String ENDING_PDF = "pdf";
 | 
			
		||||
   public static final String ENDING_TIFF = "tiff";
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   private FabianUtil() {}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   public static class ProgressHandler {
 | 
			
		||||
 | 
			
		||||
      private static final DecimalFormat TIME_DECIMAL_FORMAT = new DecimalFormat("00");
 | 
			
		||||
      protected static final int DEFAULT_LOG_DISTANCE = 50;
 | 
			
		||||
      protected static final int DEFAULT_TIME_LOG_DISTANCE = 15;
 | 
			
		||||
 | 
			
		||||
      private final ZonedDateTime startTime;
 | 
			
		||||
      private final BigDecimal logDistance;
 | 
			
		||||
      private final int timeLogDistance;
 | 
			
		||||
      private BigDecimal counter = BigDecimal.ZERO;
 | 
			
		||||
      private ZonedDateTime lastLoggedAt;
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
      public ProgressHandler() {
 | 
			
		||||
         this(DEFAULT_LOG_DISTANCE, DEFAULT_TIME_LOG_DISTANCE);
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
      public ProgressHandler(int logDistance) {
 | 
			
		||||
         this(logDistance, DEFAULT_TIME_LOG_DISTANCE);
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
      public ProgressHandler(int logDistance, int timeLogDistance) {
 | 
			
		||||
         startTime = ZonedDateTime.now();
 | 
			
		||||
         this.logDistance = validateBD(logDistance, "Log-Distanz");
 | 
			
		||||
         this.timeLogDistance = validateInt(timeLogDistance, "Zeitbasierte Log-Distanz");
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
      protected BigDecimal validateBD(int bd, String name) {
 | 
			
		||||
         return BigDecimal.valueOf(validateInt(bd, name));
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
      protected int validateInt(int n, String name) {
 | 
			
		||||
         if (n < 1) {
 | 
			
		||||
            throw new RuntimeException(name + " darf nicht 0 oder negativ sein.");
 | 
			
		||||
         }
 | 
			
		||||
         return n;
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
      protected synchronized BigDecimal getCounter() {
 | 
			
		||||
         return counter;
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
      protected synchronized int getTimeLogDistance() {
 | 
			
		||||
         return timeLogDistance;
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
      public synchronized void tickAndLog() {
 | 
			
		||||
         tick();
 | 
			
		||||
         logIf();
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
      protected synchronized void tick() {
 | 
			
		||||
         counter = counter.add(BigDecimal.ONE);
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
      public synchronized void logAndTick() {
 | 
			
		||||
         logIf();
 | 
			
		||||
         tick();
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
      public synchronized void logIf() {
 | 
			
		||||
         if (shouldLog()) {
 | 
			
		||||
            log();
 | 
			
		||||
         }
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
      protected synchronized boolean shouldLog() {
 | 
			
		||||
         return counter.remainder(logDistance).signum() == 0 || isBored();
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
      private synchronized boolean isBored() {
 | 
			
		||||
         return getTimeSinceLastLogged() >= timeLogDistance;
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
      protected synchronized long getTimeSinceLastLogged() {
 | 
			
		||||
         ZonedDateTime time = lastLoggedAt == null ? startTime : lastLoggedAt;
 | 
			
		||||
         return time.until(ZonedDateTime.now(), ChronoUnit.SECONDS);
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
      public synchronized void log() {
 | 
			
		||||
         Duration timePassed = getTimePassed();
 | 
			
		||||
         System.out.println(counter + " Stück erledigt. " + formatDuration(timePassed) + " vergangen.");
 | 
			
		||||
         updateLastLoggedAt();
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
      protected synchronized void updateLastLoggedAt() {
 | 
			
		||||
         lastLoggedAt = ZonedDateTime.now();
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
      protected synchronized Duration getTimePassed() {
 | 
			
		||||
         return Duration.ofSeconds(startTime.until(ZonedDateTime.now(), ChronoUnit.SECONDS));
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
      protected String formatDuration(Duration dur) {
 | 
			
		||||
         return formatNumber(dur.toHours()) + ":" + formatNumber(dur.toMinutesPart()) + ":" + formatNumber(dur.toSecondsPart());
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
      protected String formatNumber(Number n) {
 | 
			
		||||
         return TIME_DECIMAL_FORMAT.format(n == null ? 0 : n);
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
      public synchronized void logFinal() {
 | 
			
		||||
         Duration timePassed = getTimePassed();
 | 
			
		||||
         System.out.println(counter + " Stück (100 %) erledigt. " + formatDuration(timePassed) + " vergangen.");
 | 
			
		||||
      }
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   public static class ProgressHandlerWithTotal extends ProgressHandler {
 | 
			
		||||
 | 
			
		||||
      private static final DecimalFormat PERCENTAGE_FORMAT = new DecimalFormat("00.00");
 | 
			
		||||
 | 
			
		||||
      private final BigDecimal total;
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
      public ProgressHandlerWithTotal(int total) {
 | 
			
		||||
         this(DEFAULT_LOG_DISTANCE, DEFAULT_TIME_LOG_DISTANCE, total);
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
      public ProgressHandlerWithTotal(int logDistance, int total) {
 | 
			
		||||
         this(logDistance, DEFAULT_TIME_LOG_DISTANCE, total);
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
      public ProgressHandlerWithTotal(int logDistance, int timeLogDistance, int total) {
 | 
			
		||||
         super(logDistance, timeLogDistance);
 | 
			
		||||
         this.total = validateBD(total, "Gesamt-Anzahl");
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
      @Override
 | 
			
		||||
      protected synchronized boolean shouldLog() {
 | 
			
		||||
         return super.shouldLog() && getCounter().compareTo(total) <= 0;
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
      @Override
 | 
			
		||||
      public synchronized void log() {
 | 
			
		||||
         BigDecimal counter = getCounter();
 | 
			
		||||
         BigDecimal percentage = BigDecimal.valueOf(100).divide(total, MathContext.DECIMAL64).multiply(counter);
 | 
			
		||||
         Duration timePassed = getTimePassed();
 | 
			
		||||
 | 
			
		||||
         String remainingTimePart = "";
 | 
			
		||||
         if (counter.signum() != 0) {
 | 
			
		||||
            BigDecimal secondsPassed = BigDecimal.valueOf(timePassed.getSeconds());
 | 
			
		||||
            BigDecimal totalSeconds = secondsPassed.divide(counter, MathContext.DECIMAL64).multiply(total);
 | 
			
		||||
            BigDecimal secondsToGo = totalSeconds.subtract(secondsPassed);
 | 
			
		||||
            Duration timeToGo = Duration.of(secondsToGo.longValue(), ChronoUnit.SECONDS);
 | 
			
		||||
            remainingTimePart = ", " + formatDuration(timeToGo) + " verbleibend";
 | 
			
		||||
         }
 | 
			
		||||
 | 
			
		||||
         String counterPrint = total.compareTo(BigDecimal.TEN) >= 0 ? formatNumber(counter) : counter.toString();
 | 
			
		||||
         System.out.println(counterPrint + " Stück von " + total + " (" + PERCENTAGE_FORMAT.format(percentage) + " %) erledigt. "
 | 
			
		||||
             + formatDuration(timePassed) + " vergangen" + remainingTimePart + ".");
 | 
			
		||||
         updateLastLoggedAt();
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
      public void logContinuouslyTimeBased() {
 | 
			
		||||
         new Thread(this::logTimeBased).start();
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
      private synchronized void logTimeBased() {
 | 
			
		||||
         while (getCounter().compareTo(total) != 0) {
 | 
			
		||||
            logIf();
 | 
			
		||||
            long timeSinceLastLogged = getTimeSinceLastLogged();
 | 
			
		||||
            try {
 | 
			
		||||
               TimeUnit.SECONDS.sleep(timeSinceLastLogged - getTimeLogDistance());
 | 
			
		||||
            } catch (InterruptedException ie) {
 | 
			
		||||
               Thread.currentThread().interrupt();
 | 
			
		||||
               return;
 | 
			
		||||
            }
 | 
			
		||||
         }
 | 
			
		||||
      }
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   public static void writeTestOutput(String filename, String ending, String content) {
 | 
			
		||||
      writeTestOutput(filename, ending, strToBytes(content));
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   public static byte[] strToBytes(String str) {
 | 
			
		||||
      return str.getBytes(StandardCharsets.UTF_8);
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   public static void writeTestOutput(String filename, String ending, byte[] content) {
 | 
			
		||||
      writeToFile(STD_DIR_TEMP + "testOutput/", filename, ending, content);
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   public static void writeDump(String ending, String content) {
 | 
			
		||||
      writeDump(ending, strToBytes(content));
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   public static void writeDump(String ending, byte[] content) {
 | 
			
		||||
      writeToFile(STD_DIR_TEMP, "dump", ending, content);
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   public static void writeDump(String filename, String ending, String content) {
 | 
			
		||||
      writeDump(filename, ending, strToBytes(content));
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   public static void writeDump(String filename, String ending, byte[] content) {
 | 
			
		||||
      writeToFile(STD_DIR_TEMP, filename, ending, content);
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   public static void writeToFile(String path, String filename, String ending, byte[] content) {
 | 
			
		||||
      try {
 | 
			
		||||
         Files.write(Paths.get(path + filename + "." + ending), content);
 | 
			
		||||
      } catch (IOException e) {
 | 
			
		||||
         throw new RuntimeException(e);
 | 
			
		||||
      }
 | 
			
		||||
   }
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										164
									
								
								Java11.java
									
									
									
									
									
								
							
							
						
						
									
										164
									
								
								Java11.java
									
									
									
									
									
								
							@@ -1,164 +0,0 @@
 | 
			
		||||
import java.awt.Point;
 | 
			
		||||
import java.io.IOException;
 | 
			
		||||
import java.lang.annotation.Documented;
 | 
			
		||||
import java.lang.annotation.ElementType;
 | 
			
		||||
import java.lang.annotation.Retention;
 | 
			
		||||
import java.lang.annotation.RetentionPolicy;
 | 
			
		||||
import java.lang.annotation.Target;
 | 
			
		||||
import java.nio.file.Files;
 | 
			
		||||
import java.nio.file.Path;
 | 
			
		||||
import java.util.*;
 | 
			
		||||
import java.util.function.Predicate;
 | 
			
		||||
import java.util.stream.Collectors;
 | 
			
		||||
 | 
			
		||||
class Java11 {
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   public static void main(String[] args) {
 | 
			
		||||
      ////////
 | 
			
		||||
      // 1. //
 | 
			
		||||
      ////////
 | 
			
		||||
      // if the resource is referenced by a final or effectively final variable,
 | 
			
		||||
      // a try-with-resources statement can manage a resource without a new variable being declared:
 | 
			
		||||
      MyAutoCloseable mac = new MyAutoCloseable();
 | 
			
		||||
      try (mac) {
 | 
			
		||||
         // do some stuff with mac
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
      ////////
 | 
			
		||||
      // 2. //
 | 
			
		||||
      ////////
 | 
			
		||||
      // use diamond operator in conjunction with anonymous inner classes
 | 
			
		||||
      List<Integer> fc = new ArrayList<>(1) {}; // anonymous inner class
 | 
			
		||||
 | 
			
		||||
      List<? extends Integer> fc0 = new ArrayList<>(1) {
 | 
			
		||||
         // anonymous inner class
 | 
			
		||||
      };
 | 
			
		||||
 | 
			
		||||
      List<?> fc1 = new ArrayList<>(1) {}; // anonymous inner class
 | 
			
		||||
 | 
			
		||||
      ////////
 | 
			
		||||
      // 4. //
 | 
			
		||||
      ////////
 | 
			
		||||
      // Local Variable Type Inference
 | 
			
		||||
      var message = "Hello, Java 10";
 | 
			
		||||
      var point = new Point(1, 2);
 | 
			
		||||
 | 
			
		||||
      ////////
 | 
			
		||||
      // 5. //
 | 
			
		||||
      ////////
 | 
			
		||||
      // immutable Collections/ Maps
 | 
			
		||||
      Map<String, Integer> meineMap = Map.of();
 | 
			
		||||
      Set<String> meinSet = Set.of("key1", "key2", "key3");
 | 
			
		||||
      List<String> meineListe = List.copyOf(meinSet);
 | 
			
		||||
 | 
			
		||||
      ////////
 | 
			
		||||
      // 6. //
 | 
			
		||||
      ////////
 | 
			
		||||
      // Optional to Stream Optional.stream() gives us an easy way to you use the power of Streams on Optional elements
 | 
			
		||||
      List<String> filteredList = new ArrayList<Optional<String>>().stream()
 | 
			
		||||
         .flatMap(Optional::stream)
 | 
			
		||||
         .collect(Collectors.toList());
 | 
			
		||||
 | 
			
		||||
      ////////
 | 
			
		||||
      // 7. //
 | 
			
		||||
      ////////
 | 
			
		||||
      // Collectors get additional methods to collect a Stream into unmodifiable List, Map or Set:
 | 
			
		||||
      List<Integer> evenList = new ArrayList<Integer>().stream()
 | 
			
		||||
         .filter(i -> i % 2 == 0)
 | 
			
		||||
         .collect(Collectors.toUnmodifiableList());
 | 
			
		||||
 | 
			
		||||
      ////////
 | 
			
		||||
      // 8. //
 | 
			
		||||
      ////////
 | 
			
		||||
      // Optional got a new method orElseThrow() which doesn’t take any argument and throws NoSuchElementException if no value is present:
 | 
			
		||||
      Integer firstEven = List.of(1, 2, 3).stream()
 | 
			
		||||
         .filter(i -> i % 2 == 0)
 | 
			
		||||
         .findFirst()
 | 
			
		||||
         .orElseThrow();
 | 
			
		||||
 | 
			
		||||
      ////////
 | 
			
		||||
      // 9. //
 | 
			
		||||
      ////////
 | 
			
		||||
      // Java 11 adds a few new methods to the String class
 | 
			
		||||
      var multilineString = "Baeldung helps \n \n developers \n explore Java.";
 | 
			
		||||
      List<String> lines = multilineString.lines().collect(Collectors.toList());
 | 
			
		||||
      assert "    ".isBlank();
 | 
			
		||||
      assert "    x    ".strip().equals("x");
 | 
			
		||||
      assert "    x    ".stripLeading().equals("x    ");
 | 
			
		||||
      assert "    x    ".stripTrailing().equals("    x");
 | 
			
		||||
      assert "x".repeat(5).equals("xxxxx");
 | 
			
		||||
 | 
			
		||||
      /////////
 | 
			
		||||
      // 10. //
 | 
			
		||||
      /////////
 | 
			
		||||
      // We can use the new readString and writeString static methods from the Files class:
 | 
			
		||||
      String fileContent = null;
 | 
			
		||||
      try {
 | 
			
		||||
         Path filePath = Files.writeString(Files.createTempFile("demo", ".txt"), "Sample text");
 | 
			
		||||
         fileContent = Files.readString(filePath);
 | 
			
		||||
      } catch (IOException ioe) {}
 | 
			
		||||
      assert fileContent.equals("Sample text");
 | 
			
		||||
 | 
			
		||||
      /////////
 | 
			
		||||
      // 11. //
 | 
			
		||||
      /////////
 | 
			
		||||
      // The java.util.Collection interface contains a new default toArray method which takes an IntFunction argument.
 | 
			
		||||
      List<String> sampleList = List.of("Java", "Kotlin");
 | 
			
		||||
      String[] sampleArray = sampleList.toArray(String[]::new);
 | 
			
		||||
 | 
			
		||||
      /////////
 | 
			
		||||
      // 12. //
 | 
			
		||||
      /////////
 | 
			
		||||
      // A static not method has been added to the Predicate interface.
 | 
			
		||||
      sampleList = List.of("Java", "\n \n", "Kotlin", " ");
 | 
			
		||||
      List<String> withoutBlanks = sampleList.stream()
 | 
			
		||||
         .filter(Predicate.not(String::isBlank))
 | 
			
		||||
         .collect(Collectors.toList());
 | 
			
		||||
      // While not(isBlank) reads more naturally than isBlank.negate(), the big advantage is that we can also use not with method references,
 | 
			
		||||
      // like not(String::isBlank).
 | 
			
		||||
 | 
			
		||||
      /////////
 | 
			
		||||
      // 13. //
 | 
			
		||||
      /////////
 | 
			
		||||
      // Support for using the local variable syntax (var keyword) in lambda parameters was added in Java 11.
 | 
			
		||||
      // We can make use of this feature to apply modifiers to our local variables, like defining a type annotation:
 | 
			
		||||
      sampleList = List.of("Java", "Kotlin");
 | 
			
		||||
      String resultString = sampleList.stream()
 | 
			
		||||
         .map((@Nonnull var x) -> x.toUpperCase())
 | 
			
		||||
         .collect(Collectors.joining(", "));
 | 
			
		||||
      assert resultString.equals("JAVA, KOTLIN");
 | 
			
		||||
   }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
class MyAutoCloseable implements AutoCloseable {
 | 
			
		||||
   public void close() {}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
////////
 | 
			
		||||
// 3. //
 | 
			
		||||
////////
 | 
			
		||||
// Interfaces in the upcoming JVM version can have private methods, which can be used to split lengthy default methods
 | 
			
		||||
interface InterfaceWithPrivateMethods {
 | 
			
		||||
   private static String staticPrivate() {
 | 
			
		||||
      return "static private";
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
   private String instancePrivate() {
 | 
			
		||||
      return "instance private";
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
   default void check() {
 | 
			
		||||
      String result = staticPrivate();
 | 
			
		||||
      InterfaceWithPrivateMethods pvt = new InterfaceWithPrivateMethods() {
 | 
			
		||||
         // anonymous class
 | 
			
		||||
      };
 | 
			
		||||
      result = pvt.instancePrivate();
 | 
			
		||||
   }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@Documented
 | 
			
		||||
@Retention(RetentionPolicy.RUNTIME)
 | 
			
		||||
@Target(ElementType.PARAMETER)
 | 
			
		||||
@interface Nonnull {}
 | 
			
		||||
							
								
								
									
										502
									
								
								LoadingBar.java
									
									
									
									
									
								
							
							
						
						
									
										502
									
								
								LoadingBar.java
									
									
									
									
									
								
							@@ -1,502 +0,0 @@
 | 
			
		||||
import java.io.BufferedReader;
 | 
			
		||||
import java.io.InputStreamReader;
 | 
			
		||||
import java.io.IOException;
 | 
			
		||||
import java.math.BigDecimal;
 | 
			
		||||
import java.math.MathContext;
 | 
			
		||||
import java.math.RoundingMode;
 | 
			
		||||
import java.nio.charset.StandardCharsets;
 | 
			
		||||
import java.text.DecimalFormat;
 | 
			
		||||
import java.time.format.DateTimeFormatter;
 | 
			
		||||
import java.time.LocalTime;
 | 
			
		||||
import java.time.temporal.ChronoUnit;
 | 
			
		||||
import java.util.Arrays;
 | 
			
		||||
import java.util.concurrent.TimeUnit;
 | 
			
		||||
import java.util.List;
 | 
			
		||||
import java.util.Objects;
 | 
			
		||||
import java.util.regex.Pattern;
 | 
			
		||||
import java.util.stream.Collectors;
 | 
			
		||||
 | 
			
		||||
public class LoadingBar {
 | 
			
		||||
 | 
			
		||||
   private static final String TIME_FORMAT = "HH:mm";
 | 
			
		||||
   private static final DateTimeFormatter TIME_FORMATTER = DateTimeFormatter.ofPattern(TIME_FORMAT);
 | 
			
		||||
   private static final Pattern TIME_PATTERN = Pattern.compile("(?>[01]\\d|2[0-4]):[0-5]\\d");
 | 
			
		||||
   private static final Pattern LUNCH_DURATION_PATTERN = Pattern.compile("\\d+");
 | 
			
		||||
   private static final Pattern OFFSET_PATTERN = Pattern.compile("[+-]\\d+");
 | 
			
		||||
   private static final DecimalFormat PERCENTAGE_FORMAT = new DecimalFormat("00.00");
 | 
			
		||||
   private static final int MIN_LUNCH_DURATION = 30;
 | 
			
		||||
   private static final LocalTime LATEST_LUNCH_TIME = LocalTime.of(13, 30);
 | 
			
		||||
   private static final int MINS_PER_HOUR = 60;
 | 
			
		||||
   private static final BigDecimal MINS_PER_HOUR_BD = BigDecimal.valueOf(MINS_PER_HOUR);
 | 
			
		||||
   private static final long DEFAULT_NUMBER_WORK_MINS_BEFORE_LUNCH = 5L * MINS_PER_HOUR;
 | 
			
		||||
   private static final int MAX_NUMBER_WORK_MINS_WITHOUT_LUNCH = 6 * MINS_PER_HOUR;
 | 
			
		||||
   private static final long MAX_NUMBER_WORK_MINS = 8L * MINS_PER_HOUR;
 | 
			
		||||
   private static final int LINE_LENGTH = 100;
 | 
			
		||||
   private static final MathContext MC_INTEGER = new MathContext(1, RoundingMode.HALF_EVEN);
 | 
			
		||||
   private static final BigDecimal ONE_HUNDRED_PERCENT = BigDecimal.valueOf(100);
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   private enum DaySection {
 | 
			
		||||
      MITTAG("-m", "Mittag"),
 | 
			
		||||
      ZAPFENSTREICH("-z", "Zapfenstreich");
 | 
			
		||||
 | 
			
		||||
      private final String param;
 | 
			
		||||
      private final String description;
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
      DaySection(String param, String description) {
 | 
			
		||||
         this.param = param;
 | 
			
		||||
         this.description = description;
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
      public static DaySection byParam(String param) {
 | 
			
		||||
         return Arrays.stream(DaySection.values()).filter((DaySection section) -> section.getParam().equals(param)).findFirst().orElse(null);
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
      public String getParam() {
 | 
			
		||||
         return param;
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
      public String getDescription() {
 | 
			
		||||
         return description;
 | 
			
		||||
      }
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   private final LocalTime startTime;
 | 
			
		||||
   private LocalTime endTime;
 | 
			
		||||
   private long totalMinutes;
 | 
			
		||||
   private BigDecimal totalMinutesBD;
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   private LoadingBar(LocalTime startTime) {
 | 
			
		||||
      this.startTime = startTime;
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   public static void main(String[] args) throws IOException {
 | 
			
		||||
      if (args.length == 0) {
 | 
			
		||||
         askParametersAndRun();
 | 
			
		||||
      } else {
 | 
			
		||||
         parseParametersAndRun(args);
 | 
			
		||||
      }
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   private static void askParametersAndRun() throws IOException {
 | 
			
		||||
      var br = new BufferedReader(new InputStreamReader(System.in, StandardCharsets.UTF_8));
 | 
			
		||||
      print("Ankunftszeit: ");
 | 
			
		||||
      var startTime = LocalTime.parse(br.readLine(), TIME_FORMATTER).truncatedTo(ChronoUnit.MINUTES);
 | 
			
		||||
      var lb = new LoadingBar(startTime);
 | 
			
		||||
      if (lb.hasMittagspauseArrived()) {
 | 
			
		||||
         handleMittagspause(br, lb);
 | 
			
		||||
         lb.showLoadingBar();
 | 
			
		||||
      }
 | 
			
		||||
      handleZapfenstreich(br, lb);
 | 
			
		||||
      lb.showLoadingBar();
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   private static void println(Object o) {
 | 
			
		||||
      System.out.println(o);
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   private static void print(Object o) {
 | 
			
		||||
      System.out.print(o);
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   private static void handleMittagspause(BufferedReader br, LoadingBar lb) throws IOException {
 | 
			
		||||
      print("Mittagspause verschieben um (optional): ");
 | 
			
		||||
      String mittagspauseOffsetRaw = br.readLine();
 | 
			
		||||
      if (mittagspauseOffsetRaw != null && !mittagspauseOffsetRaw.isBlank()) {
 | 
			
		||||
         var mittagspauseOffset = Integer.parseInt(mittagspauseOffsetRaw);
 | 
			
		||||
         lb.initMittagspause(mittagspauseOffset);
 | 
			
		||||
         return;
 | 
			
		||||
      }
 | 
			
		||||
      print("Mittagspause um (optional): ");
 | 
			
		||||
      String manualMittagspauseRaw = br.readLine();
 | 
			
		||||
      if (manualMittagspauseRaw != null && !manualMittagspauseRaw.isBlank()) {
 | 
			
		||||
         var manualMittagspause = LocalTime.parse(manualMittagspauseRaw, TIME_FORMATTER).truncatedTo(ChronoUnit.MINUTES);
 | 
			
		||||
         lb.initMittagspause(manualMittagspause);
 | 
			
		||||
      } else {
 | 
			
		||||
         lb.initMittagspause();
 | 
			
		||||
      }
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   private static void handleZapfenstreich(BufferedReader br, LoadingBar lb) throws IOException {
 | 
			
		||||
      print("Mittagspause hat gedauert (optional): ");
 | 
			
		||||
      String mittagspauseDurationRaw = br.readLine();
 | 
			
		||||
      Integer mittagspauseDuration = null;
 | 
			
		||||
      if (mittagspauseDurationRaw != null && !mittagspauseDurationRaw.isBlank()) {
 | 
			
		||||
         mittagspauseDuration = Integer.valueOf(mittagspauseDurationRaw);
 | 
			
		||||
      }
 | 
			
		||||
      LocalTime vorlaeufigeEndzeit = lb.startTime.plusMinutes(MAX_NUMBER_WORK_MINS)
 | 
			
		||||
         .plusMinutes(mittagspauseDuration != null ? mittagspauseDuration : MIN_LUNCH_DURATION);
 | 
			
		||||
      println("Endzeit: " + TIME_FORMATTER.format(vorlaeufigeEndzeit));
 | 
			
		||||
      print("Feierabend verschieben um (optional): ");
 | 
			
		||||
      String zapfenstreichOffsetRaw = br.readLine();
 | 
			
		||||
      Integer zapfenstreichOffset = null;
 | 
			
		||||
      if (zapfenstreichOffsetRaw != null && !zapfenstreichOffsetRaw.isBlank()) {
 | 
			
		||||
         zapfenstreichOffset = Integer.valueOf(zapfenstreichOffsetRaw);
 | 
			
		||||
         lb.initZapfenstreich(mittagspauseDuration, zapfenstreichOffset);
 | 
			
		||||
         return;
 | 
			
		||||
      }
 | 
			
		||||
      print("Manuelle Uhrzeit Feierabend (optional): ");
 | 
			
		||||
      String manualZapfenstreichRaw = br.readLine();
 | 
			
		||||
      LocalTime manualZapfenstreich = null;
 | 
			
		||||
      if (manualZapfenstreichRaw != null && !manualZapfenstreichRaw.isBlank()) {
 | 
			
		||||
         manualZapfenstreich = LocalTime.parse(manualZapfenstreichRaw, TIME_FORMATTER).truncatedTo(ChronoUnit.MINUTES);
 | 
			
		||||
         lb.initZapfenstreich(mittagspauseDuration, manualZapfenstreich);
 | 
			
		||||
      } else {
 | 
			
		||||
         lb.initZapfenstreich(mittagspauseDuration);
 | 
			
		||||
      }
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   private static void parseParametersAndRun(String[] args) {
 | 
			
		||||
      LoadingBar lb = getLoadingBarFromCLI(args);
 | 
			
		||||
      lb.showLoadingBar();
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   private static LoadingBar getLoadingBarFromCLI(String[] args) {
 | 
			
		||||
      String nextArg = args[0];
 | 
			
		||||
      if ("--help".equals(nextArg)) {
 | 
			
		||||
         printHelp();
 | 
			
		||||
         System.exit(1);
 | 
			
		||||
      }
 | 
			
		||||
      verifyMinimumNumberOfArgs(args);
 | 
			
		||||
      verifyTimeFormat(nextArg, "Erstes Argument");
 | 
			
		||||
      var startTime = LocalTime.parse(nextArg, TIME_FORMATTER).truncatedTo(ChronoUnit.MINUTES);
 | 
			
		||||
      nextArg = args[1];
 | 
			
		||||
      var section = DaySection.byParam(nextArg);
 | 
			
		||||
      verifyDaySection(section, nextArg);
 | 
			
		||||
      return section == DaySection.MITTAG ? getLoadingBarMittagspause(args, startTime) : getLoadingBarZapfenstreich(args, startTime);
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   private static LoadingBar getLoadingBarMittagspause(String[] args, LocalTime startTime) {
 | 
			
		||||
      var lb = new LoadingBar(startTime);
 | 
			
		||||
      if (args.length == 2) {
 | 
			
		||||
         lb.initMittagspause();
 | 
			
		||||
         return lb;
 | 
			
		||||
      }
 | 
			
		||||
      String nextArg = args[2];
 | 
			
		||||
      if (OFFSET_PATTERN.matcher(nextArg).matches()) {
 | 
			
		||||
         lb.initMittagspause(Integer.parseInt(nextArg));
 | 
			
		||||
         return lb;
 | 
			
		||||
      }
 | 
			
		||||
      verifyTimeFormat(nextArg, "Argument nach " + DaySection.MITTAG.getParam());
 | 
			
		||||
      var manualMittagspause = LocalTime.parse(nextArg, TIME_FORMATTER).truncatedTo(ChronoUnit.MINUTES);
 | 
			
		||||
      lb.initMittagspause(manualMittagspause);
 | 
			
		||||
      return lb;
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   private static LoadingBar getLoadingBarZapfenstreich(String[] args, LocalTime startTime) {
 | 
			
		||||
      var lb = new LoadingBar(startTime);
 | 
			
		||||
      if (args.length == 2) {
 | 
			
		||||
         lb.initZapfenstreich();
 | 
			
		||||
         return lb;
 | 
			
		||||
      }
 | 
			
		||||
      String nextArg = args[2];
 | 
			
		||||
      LocalTime maxZapfenstreich = null;
 | 
			
		||||
      int endTimeOffset = 0;
 | 
			
		||||
      Integer lunchDuration = null;
 | 
			
		||||
      if (TIME_PATTERN.matcher(nextArg).matches()) {
 | 
			
		||||
         maxZapfenstreich = LocalTime.parse(nextArg, TIME_FORMATTER).truncatedTo(ChronoUnit.MINUTES);
 | 
			
		||||
      } else if (OFFSET_PATTERN.matcher(nextArg).matches()) {
 | 
			
		||||
         endTimeOffset = Integer.parseInt(nextArg);
 | 
			
		||||
      } else {
 | 
			
		||||
         verifyDurationFormat(nextArg, "Argument nach " + DaySection.ZAPFENSTREICH.getParam());
 | 
			
		||||
         lunchDuration = Integer.parseInt(nextArg);
 | 
			
		||||
      }
 | 
			
		||||
      if (args.length == 3) {
 | 
			
		||||
         if (maxZapfenstreich == null && endTimeOffset == 0) {
 | 
			
		||||
            lb.initZapfenstreich(lunchDuration);
 | 
			
		||||
         } else if (maxZapfenstreich == null) {
 | 
			
		||||
            lb.initZapfenstreich(lunchDuration, endTimeOffset);
 | 
			
		||||
         } else {
 | 
			
		||||
            lb.initZapfenstreich(lunchDuration, maxZapfenstreich);
 | 
			
		||||
         }
 | 
			
		||||
         return lb;
 | 
			
		||||
      }
 | 
			
		||||
      nextArg = args[3];
 | 
			
		||||
      if (lunchDuration == null) {
 | 
			
		||||
         println("Letztes Argument darf nur auf Mittagspausendauer folgen.");
 | 
			
		||||
         System.exit(1);
 | 
			
		||||
      }
 | 
			
		||||
      if (maxZapfenstreich == null && !OFFSET_PATTERN.matcher(nextArg).matches()) {
 | 
			
		||||
         verifyTimeFormat(nextArg, "Letztes Argument nach " + DaySection.ZAPFENSTREICH.getParam() + " und Mittagspausendauer");
 | 
			
		||||
         maxZapfenstreich = LocalTime.parse(nextArg, TIME_FORMATTER).truncatedTo(ChronoUnit.MINUTES);
 | 
			
		||||
         lb.initZapfenstreich(lunchDuration, maxZapfenstreich);
 | 
			
		||||
         return lb;
 | 
			
		||||
      }
 | 
			
		||||
      verifyOffsetFormat(nextArg, "Letztes Argument nach " + DaySection.ZAPFENSTREICH.getParam() + " und Enduhrzeit");
 | 
			
		||||
      endTimeOffset = Integer.parseInt(nextArg);
 | 
			
		||||
      lb.initZapfenstreich(lunchDuration, endTimeOffset);
 | 
			
		||||
      return lb;
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   private static void verifyMinimumNumberOfArgs(String[] args) {
 | 
			
		||||
      if (args.length >= 2) {
 | 
			
		||||
         return;
 | 
			
		||||
      }
 | 
			
		||||
      println("Mindestens 2 Argumente müssen gegeben sein.");
 | 
			
		||||
      printHelp();
 | 
			
		||||
      System.exit(1);
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   private static void verifyTimeFormat(String param, String errMsgPrefix) {
 | 
			
		||||
      verifyInputFormat(TIME_PATTERN, param, errMsgPrefix, "Uhrzeitformat (" + TIME_FORMAT + ")", false);
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   private static void verifyDurationFormat(String param, String errMsgPrefix) {
 | 
			
		||||
      verifyInputFormat(LUNCH_DURATION_PATTERN, param, errMsgPrefix, "Minutenanzahl (ganze Zahl)", true);
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   private static void verifyOffsetFormat(String param, String errMsgPrefix) {
 | 
			
		||||
      verifyInputFormat(OFFSET_PATTERN, param, errMsgPrefix, "Minutendifferenz (ganze Zahl mit Vorzeichen)", false);
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   private static void verifyInputFormat(Pattern pattern, String param, String errMsgPrefix, String firstInputPart, boolean timeInputPossible) {
 | 
			
		||||
      if (pattern.matcher(param).matches()) {
 | 
			
		||||
         return;
 | 
			
		||||
      }
 | 
			
		||||
      var possibleTimeInputPart = timeInputPossible ? " oder Uhrzeitformat (" + TIME_FORMAT + ")" : "";
 | 
			
		||||
      println(errMsgPrefix + " \"" + param + "\" muss " + firstInputPart + possibleTimeInputPart + " entsprechen.");
 | 
			
		||||
      System.exit(1);
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   private static void verifyDaySection(DaySection section, String param) {
 | 
			
		||||
      if (section != null) {
 | 
			
		||||
         return;
 | 
			
		||||
      }
 | 
			
		||||
      List<String> sectionDescs = Arrays.stream(DaySection.values()).map((DaySection ds) -> ds.getDescription() + " (" + ds.getParam() + ")")
 | 
			
		||||
          .collect(Collectors.toList());
 | 
			
		||||
      String sectionDescsJoined = String.join(" oder ", sectionDescs);
 | 
			
		||||
      println("Argument nach Startzeit \"" + param + "\" muss Angabe für " + sectionDescsJoined + " sein.");
 | 
			
		||||
      System.exit(1);
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   private static void printHelp() {
 | 
			
		||||
      println("Mögliche Argumente für LoadingBar:\n"
 | 
			
		||||
          + "Normalfall Vormittag (Mittagspause <= " + LATEST_LUNCH_TIME + ")\n"
 | 
			
		||||
          + TIME_FORMAT + " " + DaySection.MITTAG.getParam() + "\n"
 | 
			
		||||
          + "Vormittag mit expliziter Mittagspause (<= " + LATEST_LUNCH_TIME + ")\n"
 | 
			
		||||
          + TIME_FORMAT + " " + DaySection.MITTAG.getParam() + " " + TIME_FORMAT + "\n"
 | 
			
		||||
          + "Vormittag mit abweichender Minutenanzahl Arbeitszeit\n"
 | 
			
		||||
          + TIME_FORMAT + " " + DaySection.MITTAG.getParam() + " -+mm\n"
 | 
			
		||||
          + "Normalfall Nachmittag (Mittagspause " + MIN_LUNCH_DURATION + " min)\n"
 | 
			
		||||
          + TIME_FORMAT + " " + DaySection.ZAPFENSTREICH.getParam() + "\n"
 | 
			
		||||
          + "Nachmittag mit expliziter Länge Mittagspause (Mittagspause unter " + MIN_LUNCH_DURATION + " min wird auf " + MIN_LUNCH_DURATION + " min korrigiert)\n"
 | 
			
		||||
          + TIME_FORMAT + " " + DaySection.ZAPFENSTREICH.getParam() + " mm\n"
 | 
			
		||||
          + "Nachmittag mit explizitem Feierabend (Mittagspause je nach Minimum (s.u.))\n"
 | 
			
		||||
          + TIME_FORMAT + " " + DaySection.ZAPFENSTREICH.getParam() + " " + TIME_FORMAT + "\n"
 | 
			
		||||
          + "Nachmittag mit abweichender Minutenanzahl Arbeitszeit\n"
 | 
			
		||||
          + TIME_FORMAT + " " + DaySection.ZAPFENSTREICH.getParam() + " -+mm\n"
 | 
			
		||||
          + "Nachmittag mit explizitem Feierabend u. expliziter Länge Mittagspause (Mittagspause unter Minimum (s.u.) wird auf Minimum korrigiert)\n"
 | 
			
		||||
          + TIME_FORMAT + " " + DaySection.ZAPFENSTREICH.getParam() + " mm " + TIME_FORMAT + "\n"
 | 
			
		||||
          + "Nachmittag mit explizitem Feierabend u. abweichender Minutenanzahl Arbeitszeit\n"
 | 
			
		||||
          + TIME_FORMAT + " " + DaySection.ZAPFENSTREICH.getParam() + " " + TIME_FORMAT + " -+mm\n\n"
 | 
			
		||||
          + "Mittagspause minimum in Minuten:\n"
 | 
			
		||||
          + " - bis " + MAX_NUMBER_WORK_MINS_WITHOUT_LUNCH + " min ("
 | 
			
		||||
          + MAX_NUMBER_WORK_MINS_WITHOUT_LUNCH / MINS_PER_HOUR + " std):  0\n"
 | 
			
		||||
          + " - bis " + MAX_NUMBER_WORK_MINS_WITHOUT_LUNCH + " min + "
 | 
			
		||||
          + MIN_LUNCH_DURATION + " min: Arbeitszeit - " + MAX_NUMBER_WORK_MINS_WITHOUT_LUNCH + " min\n"
 | 
			
		||||
          + " - ab  " + MAX_NUMBER_WORK_MINS_WITHOUT_LUNCH + " min + " + MIN_LUNCH_DURATION + " min: "
 | 
			
		||||
          + MIN_LUNCH_DURATION + " min\n"
 | 
			
		||||
      );
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   protected long getPassedMinutes() {
 | 
			
		||||
      return startTime.until(LocalTime.now().truncatedTo(ChronoUnit.MINUTES), ChronoUnit.MINUTES);
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   private void setEndTime(LocalTime endTime) {
 | 
			
		||||
      this.endTime = endTime;
 | 
			
		||||
      this.totalMinutes = startTime.until(endTime, ChronoUnit.MINUTES);
 | 
			
		||||
      this.totalMinutesBD = BigDecimal.valueOf(totalMinutes);
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   private boolean hasMittagspauseArrived() {
 | 
			
		||||
      return startTime.until(LocalTime.now(), ChronoUnit.MINUTES) < DEFAULT_NUMBER_WORK_MINS_BEFORE_LUNCH;
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   private void initMittagspause() {
 | 
			
		||||
      LocalTime defaultEndTime = startTime.plusMinutes(DEFAULT_NUMBER_WORK_MINS_BEFORE_LUNCH);
 | 
			
		||||
      realInitMittagspause(defaultEndTime);
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   private void initMittagspause(int endTimeOffset) {
 | 
			
		||||
      LocalTime offsetEndTime = startTime.plusMinutes(DEFAULT_NUMBER_WORK_MINS_BEFORE_LUNCH + endTimeOffset);
 | 
			
		||||
      realInitMittagspause(offsetEndTime);
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   private void initMittagspause(LocalTime manualEndTime) {
 | 
			
		||||
      LocalTime effectiveEndTime = manualEndTime != null ? manualEndTime : startTime.plusMinutes(DEFAULT_NUMBER_WORK_MINS_BEFORE_LUNCH);
 | 
			
		||||
      realInitMittagspause(effectiveEndTime);
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   private void realInitMittagspause(LocalTime theoreticalEndTime) {
 | 
			
		||||
      setEndTime(theoreticalEndTime.isAfter(LATEST_LUNCH_TIME) ? LATEST_LUNCH_TIME : theoreticalEndTime);
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   private void initZapfenstreich() {
 | 
			
		||||
      LocalTime trueEndTime = startTime.plusMinutes(MAX_NUMBER_WORK_MINS + MIN_LUNCH_DURATION);
 | 
			
		||||
      realInitZapfenstreich(MIN_LUNCH_DURATION, trueEndTime);
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   private void initZapfenstreich(Integer manualLunchDuration) {
 | 
			
		||||
      initZapfenstreich(manualLunchDuration, 0);
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   private void initZapfenstreich(Integer manualLunchDuration, int endTimeOffset) {
 | 
			
		||||
      long minLunchDuration = getMinLunchDuration(endTimeOffset);
 | 
			
		||||
      long realLunchDuration = getRealLunchDuration(manualLunchDuration, minLunchDuration);
 | 
			
		||||
      LocalTime trueEndTime = startTime.plusMinutes(MAX_NUMBER_WORK_MINS + realLunchDuration + endTimeOffset);
 | 
			
		||||
      realInitZapfenstreich(realLunchDuration, trueEndTime);
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   private void initZapfenstreich(Integer manualLunchDuration, LocalTime manualEndTime) {
 | 
			
		||||
      LocalTime trueEndTime = manualEndTime;
 | 
			
		||||
      long minLunchDuration = getMinLunchDuration(trueEndTime);
 | 
			
		||||
      long realLunchDuration = getRealLunchDuration(manualLunchDuration, minLunchDuration);
 | 
			
		||||
      if (trueEndTime == null) {
 | 
			
		||||
         trueEndTime = startTime.plusMinutes(MAX_NUMBER_WORK_MINS + realLunchDuration);
 | 
			
		||||
      }
 | 
			
		||||
      realInitZapfenstreich(realLunchDuration, trueEndTime);
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   private long getMinLunchDuration(int endTimeOffset) {
 | 
			
		||||
      if (endTimeOffset == 0) {
 | 
			
		||||
         return MIN_LUNCH_DURATION;
 | 
			
		||||
      }
 | 
			
		||||
      long totalDuration = MAX_NUMBER_WORK_MINS + endTimeOffset;
 | 
			
		||||
      return getMinLunchDuration(totalDuration);
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   private long getMinLunchDuration(LocalTime manualEndTime) {
 | 
			
		||||
      if (manualEndTime == null) {
 | 
			
		||||
         return MIN_LUNCH_DURATION;
 | 
			
		||||
      }
 | 
			
		||||
      long totalDuration = startTime.until(manualEndTime, ChronoUnit.MINUTES);
 | 
			
		||||
      return getMinLunchDuration(totalDuration);
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   private long getMinLunchDuration(long precalculatedTotalDuration) {
 | 
			
		||||
      long effectiveLunchDuration = precalculatedTotalDuration - MAX_NUMBER_WORK_MINS_WITHOUT_LUNCH;
 | 
			
		||||
      if (effectiveLunchDuration < 0) {
 | 
			
		||||
         effectiveLunchDuration = 0;
 | 
			
		||||
      }
 | 
			
		||||
      return Math.min(effectiveLunchDuration, MIN_LUNCH_DURATION);
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   private long getRealLunchDuration(Integer manualLunchDuration, long minLunchDuration) {
 | 
			
		||||
      return manualLunchDuration != null && manualLunchDuration >= minLunchDuration ? manualLunchDuration : minLunchDuration;
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   private void realInitZapfenstreich(long effectiveLunchDuration, LocalTime effectiveEndTime) {
 | 
			
		||||
      if (effectiveLunchDuration > 0) {
 | 
			
		||||
         var totalWorkTime = LocalTime.MIDNIGHT.plusMinutes(startTime.until(effectiveEndTime, ChronoUnit.MINUTES) - effectiveLunchDuration);
 | 
			
		||||
         print("Arbeitszeit: " + TIME_FORMATTER.format(totalWorkTime) + "; ");
 | 
			
		||||
      }
 | 
			
		||||
      setEndTime(effectiveEndTime);
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   private void showLoadingBar() {
 | 
			
		||||
      long passedMinutes = getPassedMinutes();
 | 
			
		||||
      // long passedMinutes = 0; // DEBUG
 | 
			
		||||
      if (passedMinutes > totalMinutes) {
 | 
			
		||||
         passedMinutes = totalMinutes;
 | 
			
		||||
      } else if (passedMinutes < 0) {
 | 
			
		||||
         var now = LocalTime.now().truncatedTo(ChronoUnit.SECONDS);
 | 
			
		||||
         println("!ACHTUNG! Startzeit \"" + startTime + ":00\" liegt in der Zukunft von jetzt an (" + now + ") gesehen.");
 | 
			
		||||
      }
 | 
			
		||||
      println(minutesToTimeString(totalMinutesBD) + " gesamt; Endzeit: " + TIME_FORMATTER.format(endTime));
 | 
			
		||||
      while (passedMinutes < totalMinutes) {
 | 
			
		||||
         print(fillLoadingBar(passedMinutes, true));
 | 
			
		||||
         waitUntilNextMinute();
 | 
			
		||||
         passedMinutes++;
 | 
			
		||||
      }
 | 
			
		||||
      println(fillLoadingBar(passedMinutes, false));
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   private void waitUntilNextMinute() {
 | 
			
		||||
      try {
 | 
			
		||||
         var now = LocalTime.now();
 | 
			
		||||
         var oneMinuteLater = now.plusMinutes(1).truncatedTo(ChronoUnit.MINUTES);
 | 
			
		||||
         /* We wait whole seconds to not make it overly complicated.
 | 
			
		||||
            That results in cut milliseconds: if we would have to wait 1 second and 526 milliseconds, we wait only 1 second.
 | 
			
		||||
            So, adjust for ignored milliseconds, add +1 second as it is better to switch between 00 and 01 as between 59 and 00 */
 | 
			
		||||
         TimeUnit.SECONDS.sleep(now.until(oneMinuteLater, ChronoUnit.SECONDS) + 1);
 | 
			
		||||
         // TimeUnit.MILLISECONDS.sleep(100L); // DEBUG
 | 
			
		||||
      } catch (InterruptedException ie) {
 | 
			
		||||
         Thread.currentThread().interrupt();
 | 
			
		||||
         throw new RuntimeException(ie);
 | 
			
		||||
      }
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   private String fillLoadingBar(long passedMinutes, boolean progressive) {
 | 
			
		||||
      var nonNegativePassedMinutes = BigDecimal.valueOf(passedMinutes < 0 ? 0 : passedMinutes);
 | 
			
		||||
      BigDecimal wholePercentage = ONE_HUNDRED_PERCENT
 | 
			
		||||
         // kind of reverse dreisatz to avoid having e.g. 99.9999 instead of 100 %
 | 
			
		||||
         .multiply(nonNegativePassedMinutes)
 | 
			
		||||
         .divide(totalMinutesBD, MathContext.DECIMAL64);
 | 
			
		||||
      int numberOfEquals = wholePercentage.intValue();
 | 
			
		||||
      var sb = new StringBuilder("[");
 | 
			
		||||
      for (int i = 0; i < LINE_LENGTH; i++) {
 | 
			
		||||
         if (i < numberOfEquals) {
 | 
			
		||||
            sb.append("=");
 | 
			
		||||
         } else {
 | 
			
		||||
            sb.append("-");
 | 
			
		||||
         }
 | 
			
		||||
      }
 | 
			
		||||
      BigDecimal remainingMinutes = totalMinutesBD.subtract(nonNegativePassedMinutes);
 | 
			
		||||
      sb.append("] ").append(PERCENTAGE_FORMAT.format(wholePercentage)).append("% ")
 | 
			
		||||
          .append(minutesToTimeString(nonNegativePassedMinutes)).append(" - ").append(minutesToTimeString(remainingMinutes));
 | 
			
		||||
      if (progressive) {
 | 
			
		||||
         sb.append("\r");
 | 
			
		||||
      }
 | 
			
		||||
      return sb.toString();
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   private String minutesToTimeString(BigDecimal minutes) {
 | 
			
		||||
      BigDecimal[] hoursAndMinutes = minutes.divideAndRemainder(MINS_PER_HOUR_BD, MC_INTEGER);
 | 
			
		||||
      return LocalTime.of(hoursAndMinutes[0].intValue(), hoursAndMinutes[1].intValue()).format(TIME_FORMATTER);
 | 
			
		||||
   }
 | 
			
		||||
}
 | 
			
		||||
@@ -1,200 +0,0 @@
 | 
			
		||||
import java.math.BigDecimal;
 | 
			
		||||
import java.math.MathContext;
 | 
			
		||||
import java.math.RoundingMode;
 | 
			
		||||
import java.text.DecimalFormat;
 | 
			
		||||
import java.time.LocalTime;
 | 
			
		||||
import java.time.format.DateTimeFormatter;
 | 
			
		||||
import java.time.temporal.ChronoUnit;
 | 
			
		||||
import java.util.Objects;
 | 
			
		||||
import java.util.concurrent.TimeUnit;
 | 
			
		||||
import java.util.regex.Pattern;
 | 
			
		||||
 | 
			
		||||
public class SimpleLoadingBar {
 | 
			
		||||
 | 
			
		||||
   private static final String TIME_FORMAT = "HH:mm";
 | 
			
		||||
   private static final DateTimeFormatter TIME_FORMATTER = DateTimeFormatter.ofPattern(TIME_FORMAT);
 | 
			
		||||
   private static final Pattern TIME_PATTERN = Pattern.compile("(?>[01]\\d|2[0-4]):[0-5]\\d");
 | 
			
		||||
   private static final DecimalFormat PERCENTAGE_FORMAT = new DecimalFormat("00.00");
 | 
			
		||||
   private static final int MINS_PER_HOUR = 60;
 | 
			
		||||
   private static final BigDecimal MINS_PER_HOUR_BD = BigDecimal.valueOf(MINS_PER_HOUR);
 | 
			
		||||
   private static final int LINE_LENGTH = 100;
 | 
			
		||||
   private static final MathContext MC_INTEGER = new MathContext(1, RoundingMode.HALF_EVEN);
 | 
			
		||||
   private static final BigDecimal ONE_HUNDRED_PERCENT = BigDecimal.valueOf(100);
 | 
			
		||||
 | 
			
		||||
   private final String title;
 | 
			
		||||
   private final LocalTime startTime;
 | 
			
		||||
   private LocalTime endTime;
 | 
			
		||||
   private long totalMinutes;
 | 
			
		||||
   private BigDecimal totalMinutesBD;
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   private SimpleLoadingBar(LocalTime startTime, LocalTime endTime, String title) {
 | 
			
		||||
      this.startTime = startTime;
 | 
			
		||||
      setEndTime(endTime);
 | 
			
		||||
      this.title = initTitle(title);
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   public static void main(String[] args) {
 | 
			
		||||
      if (args.length > 0 && Objects.equals(args[0], "--help")) {
 | 
			
		||||
         printHelp();
 | 
			
		||||
         return;
 | 
			
		||||
      }
 | 
			
		||||
      verifyMinimumNumberOfArgs(args);
 | 
			
		||||
      String nextArg = args[0];
 | 
			
		||||
      verifyTimeFormat(nextArg, "Erstes Argument");
 | 
			
		||||
      var firstTime = LocalTime.parse(nextArg, TIME_FORMATTER);
 | 
			
		||||
      LocalTime startTime = null;
 | 
			
		||||
      String title = "";
 | 
			
		||||
      LocalTime endTime = null;
 | 
			
		||||
      if (args.length > 1) {
 | 
			
		||||
         nextArg = args[1];
 | 
			
		||||
         if ("-msg".equals(nextArg)) {
 | 
			
		||||
            title = args.length > 2 ? args[2] : title;
 | 
			
		||||
         } else {
 | 
			
		||||
            verifyTimeFormat(nextArg, "Zweites Argument");
 | 
			
		||||
            endTime = LocalTime.parse(nextArg, TIME_FORMATTER);
 | 
			
		||||
         }
 | 
			
		||||
      }
 | 
			
		||||
      if (endTime == null) {
 | 
			
		||||
         startTime = LocalTime.now();
 | 
			
		||||
         endTime = firstTime;
 | 
			
		||||
      } else {
 | 
			
		||||
         startTime = firstTime;
 | 
			
		||||
      }
 | 
			
		||||
      if (args.length == 2 || !title.isBlank()) {
 | 
			
		||||
         new SimpleLoadingBar(startTime, endTime, title).showLoadingBar();
 | 
			
		||||
         return;
 | 
			
		||||
      }
 | 
			
		||||
      // if there are 3 arguments, the third will be discarded.
 | 
			
		||||
      boolean hasTitleArg = args.length > 3 && "-msg".equals(args[2]);
 | 
			
		||||
      title = hasTitleArg ? args[3] : title;
 | 
			
		||||
      new SimpleLoadingBar(startTime, endTime, title).showLoadingBar();
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   private static void println(Object o) {
 | 
			
		||||
      System.out.println(o);
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   private static void print(Object o) {
 | 
			
		||||
      System.out.print(o);
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   private static void printHelp() {
 | 
			
		||||
      println("Mögliche Argumente für LoadingBar:\n"
 | 
			
		||||
          + "Startzeit, Endzeit, Endnachricht (Optional)\n"
 | 
			
		||||
          + TIME_FORMAT + " " + TIME_FORMAT + " -msg <Nachricht>\n"
 | 
			
		||||
          + "Endzeit (Startzeit = jetzt), Endnachricht (Optional)\n"
 | 
			
		||||
          + TIME_FORMAT + " -msg <Nachricht>\n"
 | 
			
		||||
      );
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   private static void verifyMinimumNumberOfArgs(String[] args) {
 | 
			
		||||
      if (args.length >= 1) {
 | 
			
		||||
         return;
 | 
			
		||||
      }
 | 
			
		||||
      println("Mindestens 1 Argument muss gegeben sein.");
 | 
			
		||||
      printHelp();
 | 
			
		||||
      System.exit(1);
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   private static void verifyTimeFormat(String param, String errMsgPrefix) {
 | 
			
		||||
      if (TIME_PATTERN.matcher(param).matches()) {
 | 
			
		||||
         return;
 | 
			
		||||
      }
 | 
			
		||||
      println(errMsgPrefix + " \"" + param + "\" muss Uhrzeitformat (" + TIME_FORMAT + ") entsprechen.");
 | 
			
		||||
      System.exit(1);
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   protected long getPassedMinutes() {
 | 
			
		||||
      return startTime.until(LocalTime.now().truncatedTo(ChronoUnit.MINUTES), ChronoUnit.MINUTES);
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   private void setEndTime(LocalTime endTime) {
 | 
			
		||||
      this.endTime = endTime;
 | 
			
		||||
      this.totalMinutes = startTime.until(endTime, ChronoUnit.MINUTES);
 | 
			
		||||
      this.totalMinutesBD = BigDecimal.valueOf(totalMinutes);
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   private String initTitle(String inputTitle) {
 | 
			
		||||
      String fallbackTitle = "Ende! Endzeit " + TIME_FORMATTER.format(endTime) + " erreicht.";
 | 
			
		||||
      String effectiveTitle = inputTitle == null || inputTitle.isBlank() ? fallbackTitle : inputTitle;
 | 
			
		||||
      String separator = "*".repeat(effectiveTitle.length());
 | 
			
		||||
      return separator + "\n" + effectiveTitle + "\n" + separator;
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   private void showLoadingBar() {
 | 
			
		||||
      long passedMinutes = getPassedMinutes();
 | 
			
		||||
      // long passedMinutes = 0; // DEBUG
 | 
			
		||||
      if (passedMinutes > totalMinutes) {
 | 
			
		||||
         passedMinutes = totalMinutes;
 | 
			
		||||
      } else if (passedMinutes < 0) {
 | 
			
		||||
         var now = LocalTime.now().truncatedTo(ChronoUnit.SECONDS);
 | 
			
		||||
         println("!ACHTUNG! Startzeit \"" + startTime + ":00\" liegt in der Zukunft von jetzt an (" + now + ") gesehen.");
 | 
			
		||||
      }
 | 
			
		||||
      println(minutesToTimeString(totalMinutesBD) + " gesamt; Endzeit: " + TIME_FORMATTER.format(endTime));
 | 
			
		||||
      while (passedMinutes < totalMinutes) {
 | 
			
		||||
         print(fillLoadingBar(passedMinutes, true));
 | 
			
		||||
         waitUntilNextMinute();
 | 
			
		||||
         passedMinutes++;
 | 
			
		||||
      }
 | 
			
		||||
      println(fillLoadingBar(passedMinutes, false));
 | 
			
		||||
      println(title);
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   private String fillLoadingBar(long passedMinutes, boolean progressive) {
 | 
			
		||||
      var nonNegativePassedMinutes = BigDecimal.valueOf(passedMinutes < 0 ? 0 : passedMinutes);
 | 
			
		||||
      BigDecimal wholePercentage = ONE_HUNDRED_PERCENT
 | 
			
		||||
         // kind of reverse dreisatz to avoid having e.g. 99.9999 instead of 100 %
 | 
			
		||||
         .multiply(nonNegativePassedMinutes)
 | 
			
		||||
         .divide(totalMinutesBD, MathContext.DECIMAL64);
 | 
			
		||||
      int numberOfEquals = wholePercentage.intValue();
 | 
			
		||||
      var sb = new StringBuilder("[");
 | 
			
		||||
      for (int i = 0; i < LINE_LENGTH; i++) {
 | 
			
		||||
         if (i < numberOfEquals) {
 | 
			
		||||
            sb.append("=");
 | 
			
		||||
         } else {
 | 
			
		||||
            sb.append("-");
 | 
			
		||||
         }
 | 
			
		||||
      }
 | 
			
		||||
      BigDecimal remainingMinutes = totalMinutesBD.subtract(nonNegativePassedMinutes);
 | 
			
		||||
      sb.append("] ").append(PERCENTAGE_FORMAT.format(wholePercentage)).append("% ")
 | 
			
		||||
          .append(minutesToTimeString(nonNegativePassedMinutes)).append(" - ").append(minutesToTimeString(remainingMinutes));
 | 
			
		||||
      if (progressive) {
 | 
			
		||||
         sb.append("\r");
 | 
			
		||||
      }
 | 
			
		||||
      return sb.toString();
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   private String minutesToTimeString(BigDecimal minutes) {
 | 
			
		||||
      BigDecimal[] hoursAndMinutes = minutes.divideAndRemainder(MINS_PER_HOUR_BD, MC_INTEGER);
 | 
			
		||||
      return LocalTime.of(hoursAndMinutes[0].intValue(), hoursAndMinutes[1].intValue()).format(TIME_FORMATTER);
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   private void waitUntilNextMinute() {
 | 
			
		||||
      try {
 | 
			
		||||
         var now = LocalTime.now();
 | 
			
		||||
         var oneMinuteLater = now.plusMinutes(1).truncatedTo(ChronoUnit.MINUTES);
 | 
			
		||||
         /* We wait whole seconds to not make it overly complicated.
 | 
			
		||||
            That results in cut milliseconds: if we would have to wait 1 second and 526 milliseconds, we wait only 1 second.
 | 
			
		||||
            So, adjust for ignored milliseconds, add +1 second as it is better to switch between 00 and 01 as between 59 and 00 */
 | 
			
		||||
         TimeUnit.SECONDS.sleep(now.until(oneMinuteLater, ChronoUnit.SECONDS) + 1);
 | 
			
		||||
         // TimeUnit.MILLISECONDS.sleep(100L); // DEBUG
 | 
			
		||||
      } catch (InterruptedException ie) {
 | 
			
		||||
         Thread.currentThread().interrupt();
 | 
			
		||||
         throw new RuntimeException(ie);
 | 
			
		||||
      }
 | 
			
		||||
   }
 | 
			
		||||
}
 | 
			
		||||
		Reference in New Issue
	
	Block a user