Safe Haskell | Safe-Inferred |
---|---|
Language | Haskell2010 |
A progress bar in the terminal.
A progress bar conveys the progress of a task. Use a progress bar to provide a visual cue that processing is underway.
Synopsis
- data ProgressBar s
- newProgressBar :: Style s -> Double -> Progress s -> IO (ProgressBar s)
- hNewProgressBar :: Handle -> Style s -> Double -> Progress s -> IO (ProgressBar s)
- renderProgressBar :: Style s -> Progress s -> Timing -> Text
- updateProgress :: forall s. ProgressBar s -> (Progress s -> Progress s) -> IO ()
- incProgress :: ProgressBar s -> Int -> IO ()
- data Style s = Style {
- styleOpen :: !Text
- styleClose :: !Text
- styleDone :: !Char
- styleCurrent :: !Char
- styleTodo :: !Char
- stylePrefix :: Label s
- stylePostfix :: Label s
- styleWidth :: !ProgressBarWidth
- styleEscapeOpen :: EscapeCode s
- styleEscapeClose :: EscapeCode s
- styleEscapeDone :: EscapeCode s
- styleEscapeCurrent :: EscapeCode s
- styleEscapeTodo :: EscapeCode s
- styleEscapePrefix :: EscapeCode s
- styleEscapePostfix :: EscapeCode s
- styleOnComplete :: !OnComplete
- type EscapeCode s = Progress s -> Text
- data OnComplete
- defStyle :: Style s
- data ProgressBarWidth
- = ConstantWidth !Int
- | TerminalWidth !Int
- data Progress s = Progress {
- progressDone :: !Int
- progressTodo :: !Int
- progressCustom :: !s
- newtype Label s = Label {}
- data Timing = Timing {}
- msg :: Text -> Label s
- percentage :: Label s
- exact :: Label s
- elapsedTime :: (NominalDiffTime -> Text) -> Label s
- remainingTime :: (NominalDiffTime -> Text) -> Text -> Label s
- totalTime :: (NominalDiffTime -> Text) -> Text -> Label s
- renderDuration :: NominalDiffTime -> Text
Getting started
You want to perform some task which will take some time. You wish to show the progress of this task in the terminal.
- Determine the total amount of work
- Create a progress bar with
newProgressBar
For each unit of work:
3a. Perform the work
3b. Update the progress bar with
incProgress
Explore the Style
and the Label
types to see various ways in which
you can customize the progress bar.
You do not have to close the progress bar, or even finish the task. It is perfectly fine to stop half way (maybe your task throws an exception).
Just remember to avoid outputting text to the terminal while a progress bar is active. It will mess up the output a bit.
Example
Write a function which represents a unit of work. This could be a file copy operation, a network operation or some other expensive calculation. This example simply waits 1 second.
work :: IO () work = threadDelay 1000000 -- 1 second
And you define some work to be done. This could be a list of files to process or some jobs that need to be processed.
toBeDone :: [()] toBeDone = replicate 20 ()
Now create the progress bar. Use the default style and choose a maximum refresh rate of 10 Hz. The initial progress is 0 work done out of 20.
pb <-newProgressBar
defStyle
10 (Progress
0 20 ())
Start performing the work while keeping the user informed of the progress:
for_ toBeDone $ () -> do
work -- perform 1 unit of work
incProgress
pb 1 -- increment progress by 1
That's it! You get a nice animated progress bar in your terminal. It will look like this:
[==========>................................] 25%
Progress bars
data ProgressBar s Source #
A terminal progress bar.
A ProgressBar
value contains the state of a progress bar.
Create a progress bar with newProgressBar
or hNewProgressBar
.
Update a progress bar with updateProgress
or incProgress
.
Instances
NFData s => NFData (ProgressBar s) Source # | |
Defined in System.ProgressBar rnf :: ProgressBar s -> () # |
:: Style s | Visual style of the progress bar. |
-> Double | Maximum refresh rate in Hertz. |
-> Progress s | Initial progress. |
-> IO (ProgressBar s) |
Creates a progress bar.
The progress bar is drawn immediately. Update the progress bar with
updateProgress
or incProgress
. Do not output anything to your
terminal between updates. It will mess up the animation.
The progress bar is written to stderr
. Write to another handle
with hNewProgressBar
.
:: Handle | File handle on which the progress bar is drawn. Usually
you select a standard stream like |
-> Style s | Visual style of the progress bar. |
-> Double | Maximum refresh rate in Hertz. |
-> Progress s | Initial progress. |
-> IO (ProgressBar s) |
Creates a progress bar which outputs to the given handle.
See newProgressBar
.
Renders a progress bar.
>>>
let t = UTCTime (ModifiedJulianDay 0) 0
>>>
renderProgressBar defStyle (Progress 30 100 ()) (Timing t t)
"[============>..............................] 30%"
Note that this function can not use TerminalWidth
because it
doesn't use IO
. Use newProgressBar
or hNewProgressBar
to get
automatic width.
:: forall s. ProgressBar s | Progress bar to update. |
-> (Progress s -> Progress s) | Function to change the progress. |
-> IO () |
Change the progress of a progress bar.
This function is thread safe. Multiple threads may update a single progress bar at the same time.
There is a maximum refresh rate. This means that some updates might not be drawn.
:: ProgressBar s | Progress bar which needs an update. |
-> Int | Amount by which to increment the progress. |
-> IO () |
Increment the progress of an existing progress bar.
See updateProgress
for more information.
Options
Visual style of a progress bar.
The style determines how a progress bar is rendered to text.
The textual representation of a progress bar follows the following template:
<prefix><open><done><current><todo><close><postfix>
Where <done> and <todo> are repeated as often as necessary.
Consider the following progress bar
"Working [=======>.................] 30%"
This bar can be specified using the following style:
Style
{styleOpen
= "[" ,styleClose
= "]" ,styleDone
= '=' ,styleCurrent
= '>' ,styleTodo
= '.' ,stylePrefix
=msg
"Working" ,stylePostfix
=percentage
,styleWidth
=ConstantWidth
40 ,styleEscapeOpen
= constempty
,styleEscapeClose
= constempty
,styleEscapeDone
= constempty
,styleEscapeCurrent
= constempty
,styleEscapeTodo
= constempty
,styleEscapePrefix
= constempty
,styleEscapePostfix
= constempty
,styleOnComplete
=WriteNewline
}
Style | |
|
Instances
type EscapeCode s Source #
An escape code is a sequence of bytes which the terminal looks for and interprets as commands, not as character codes.
It is vital that the output of this function, when send to the terminal, does not result in characters being drawn.
data OnComplete Source #
What happens when a progress bar is finished.
WriteNewline | Write a new line when the progress bar is finished. The completed progress bar will remain visible. |
Clear | Clear the progress bar once it is finished. |
Instances
Generic OnComplete Source # | |
Defined in System.ProgressBar type Rep OnComplete :: Type -> Type # from :: OnComplete -> Rep OnComplete x # to :: Rep OnComplete x -> OnComplete # | |
NFData OnComplete Source # | |
Defined in System.ProgressBar rnf :: OnComplete -> () # | |
type Rep OnComplete Source # | |
The default style.
This style shows the progress as a percentage. It does not use any escape sequences.
Override some fields of the default instead of specifying all the
fields of a Style
record.
data ProgressBarWidth Source #
Width of progress bar in characters.
ConstantWidth !Int | A constant width. |
TerminalWidth !Int | Use the entire width of the terminal. Identical to |
Instances
Generic ProgressBarWidth Source # | |
Defined in System.ProgressBar type Rep ProgressBarWidth :: Type -> Type # from :: ProgressBarWidth -> Rep ProgressBarWidth x # to :: Rep ProgressBarWidth x -> ProgressBarWidth # | |
NFData ProgressBarWidth Source # | |
Defined in System.ProgressBar rnf :: ProgressBarWidth -> () # | |
type Rep ProgressBarWidth Source # | |
Defined in System.ProgressBar type Rep ProgressBarWidth = D1 ('MetaData "ProgressBarWidth" "System.ProgressBar" "terminal-progress-bar-0.4.2-BMlLa4E394S7aHr1iIEVHn" 'False) (C1 ('MetaCons "ConstantWidth" 'PrefixI 'False) (S1 ('MetaSel ('Nothing :: Maybe Symbol) 'NoSourceUnpackedness 'SourceStrict 'DecidedStrict) (Rec0 Int)) :+: C1 ('MetaCons "TerminalWidth" 'PrefixI 'False) (S1 ('MetaSel ('Nothing :: Maybe Symbol) 'NoSourceUnpackedness 'SourceStrict 'DecidedStrict) (Rec0 Int))) |
Progress
An amount of progress.
Progress | |
|
Labels
A label is a part of a progress bar that changes based on the progress.
Labels can be at the front (prefix) or at the back (postfix) of a progress bar.
Labels can use both the current amount of progress and the timing information to generate some text.
Instances
IsString (Label s) Source # | Every string is a label which ignores its input and just outputs that string. |
Defined in System.ProgressBar fromString :: String -> Label s # | |
Monoid (Label s) Source # | The mempty label always outputs an empty text. |
Semigroup (Label s) Source # | Combining labels combines their output. |
NFData (Label s) Source # | |
Defined in System.ProgressBar |
Timing information about a ProgressBar
.
This information is used by Label
s to calculate elapsed time, remaining time, total time, etc.
See elapsedTime
, remainingTime
and totalTime
.
Timing | |
|
msg :: Text -> Label s Source #
Static text.
The output does not depend on the input.
>>>
msg "foo" st
"foo"
percentage :: Label s Source #
Progress as a percentage.
>>>
runLabel $ percentage (Progress 30 100 ()) someTiming
" 30%"
Note: if no work is to be done (todo == 0) the percentage will be shown as 100%.
Progress as a fraction of the total amount of work.
>>>
runLabel $ exact (Progress 30 100 ()) someTiming
" 30/100"
elapsedTime :: (NominalDiffTime -> Text) -> Label s Source #
Amount of time that has elapsed.
Time starts when a progress bar is created.
The user must supply a function which actually renders the amount
of time that has elapsed. You can use renderDuration
or
formatTime
from time >= 1.9.
:: (NominalDiffTime -> Text) | |
-> Text | Alternative message when remaining time can't be calculated (yet). |
-> Label s |
Estimated remaining time.
Tells you how much longer some task is expected to take.
This label uses a simple estimation algorithm. It assumes progress is linear. To prevent nonsense results it won't estimate remaining time until at least 1 second of work has been done.
When it refuses to estimate the remaining time it will show an alternative message instead.
The user must supply a function which actually renders the amount
of time that has elapsed. Use renderDuration
or formatTime
from
the time >= 1.9 package.
:: (NominalDiffTime -> Text) | |
-> Text | Alternative message when total time can't be calculated (yet). |
-> Label s |
Estimated total time.
This label uses a simple estimation algorithm. It assumes progress is linear. To prevent nonsense results it won't estimate the total time until at least 1 second of work has been done.
When it refuses to estimate the total time it will show an alternative message instead.
The user must supply a function which actually renders the total
amount of time that a task will take. You can use renderDuration
or formatTime
from the time >= 1.9 package.
renderDuration :: NominalDiffTime -> Text Source #
Show amount of time.
renderDuration (fromInteger 42)
42
renderDuration (fromInteger $ 5 * 60 + 42)
05:42
renderDuration (fromInteger $ 8 * 60 * 60 + 5 * 60 + 42)
08:05:42
Use the time >= 1.9 package to get a formatTime function which
accepts NominalDiffTime
.