# frozen_string_literal: true # TODO: replace with ruby-progressbar dependency require 'io/console' module Bumbler module Progress @item_count = 0 @loaded_items = 0 # registry[item_name] = 123.45 @registry = {} class << self attr_reader :registry def register_item(name) # Build a blank key for the item @item_count += 1 unless @registry[name] @registry[name] = nil end def item_started(name) @curr_item = { name: name } render_progress end def item_finished(name, time) @registry[name] = time @loaded_items += 1 @prev_item = { name: name, time: time } @curr_item = nil if @curr_item && @curr_item[:name] == name render_progress end def tty_width # console_winsize: https://github.com/ruby/ruby/blob/f27eb8148f5a72bbacfebfecc7de9305471bb5c9/ext/io/console/console.c#L796 IO.console.winsize[1] end def bar(width) inner_size = width - 2 fill_size = [((@loaded_items / @item_count.to_f) * inner_size).to_i, inner_size].min fill = '#' * fill_size empty = ' ' * (inner_size - fill_size) "[#{fill}#{empty}]" end def render_progress # Do nothing if we don't have any items to load return if @item_count == 0 # Output components: # [#######################################] # (##/##) ... (####.##ms) # # Skip the current if there isn't enough room count = format('(%s/%d) ', @loaded_items.to_s.rjust(@item_count.to_s.size), @item_count) current = @curr_item ? "#{@curr_item[:name]}... " : '' prev = @prev_item ? format('%s (%sms)', @prev_item[:name], ('%.2f' % @prev_item[:time]).rjust(7)) : '' if $stdout.tty? width = tty_width print "\r\e[A\r\e[A" if @outputted_once @outputted_once = true # Align the bottom row space_for_current = width - (count.length + prev.length) # Render the progress puts bar(width) if space_for_current >= current.length puts count + current + prev.rjust(width - count.length - current.length) else puts count + prev.rjust(width - count.length) end elsif @curr_item puts format('%s %s', count, @curr_item[:name]) end end end end end