Source file tui_terminal_io.ml
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
open! Core
open Quickterface.Io
type t = { term : Notty_lwt.Term.t; window : Window.t }
let refresh_render { term; window } () =
let screen_width, screen_height = Notty_lwt.Term.size term in
let render_info = { Render_info.screen_width; screen_height } in
let image = Window.render ~render_info window in
Notty_lwt.Term.image term image
module Http_client = Cohttp_lwt_unix.Client
let input_then_add_to_log ~window_input ~log_item ({ window; _ } as t) () =
let%lwt res = window_input window ~refresh_render:(refresh_render t) () in
let%lwt () = Window.add_log_item window (log_item res) in
Lwt.return res
let input_any_key ({ window; _ } as t) () =
Window.input_any_key window ~refresh_render:(refresh_render t) ()
let input_text ?(prompt = "> ") t () =
input_then_add_to_log
~window_input:(Window.input_text ~prompt)
~log_item:(Log_item.input_text ~prompt)
t ()
let input_integer t () =
input_then_add_to_log ~window_input:Window.input_integer
~log_item:(fun n -> Log_item.input_text (string_of_int n))
t ()
let input_single_selection t options option_to_string () =
input_then_add_to_log
~window_input:(Window.input_single_selection ~options ~option_to_string)
~log_item:(fun selected_option ->
Log_item.input_text (option_to_string selected_option))
t ()
let input_single_selection_string t options () =
input_single_selection t options Fn.id ()
let input_multi_selection t options option_to_string () =
input_then_add_to_log
~window_input:(Window.input_multi_selection ~options ~option_to_string)
~log_item:(fun selected_options ->
Log_item.input_text
(selected_options
|> List.map ~f:option_to_string
|> String.concat ~sep:", "))
t ()
let input_multi_selection_string t options () =
input_multi_selection t options Fn.id ()
let input : type settings a.
_ -> (settings, a) Input.t -> settings -> unit -> a Lwt.t =
fun t -> function
| Text -> fun prompt -> input_text ?prompt t
| Integer -> fun () -> input_integer t
| Single_selection ->
fun (options, option_to_string) ->
input_single_selection t options option_to_string
| Multi_selection ->
fun (options, option_to_string) ->
input_multi_selection t options option_to_string
let then_refresh_render ~t f =
let%lwt () = f in
refresh_render t ()
let output_text ?options ({ window; _ } as t) text () =
Window.add_log_item window (Log_item.output_text ?options text)
|> then_refresh_render ~t
let output_math ?options ({ window; _ } as t) math () =
Window.add_log_item window (Log_item.output_math ?options math)
|> then_refresh_render ~t
let output_title ({ window; _ } as t) text () =
Window.set_title window text |> then_refresh_render ~t
let output : type options a.
?options:options -> _ -> (options, a) Output.t -> a -> unit -> unit Lwt.t =
fun ?options t -> function
| Text -> fun x -> output_text ?options t x
| Math -> fun x -> output_math ?options t x
| Title -> (
fun x ->
match options with
| None | Some () -> output_title t x)
let with_progress_bar ?label ({ window; _ } as t) ~maximum ~f () =
Window.with_progress_bar window
~config:{ Progress_bar_config.label; maximum_value = maximum }
~refresh_render:(refresh_render t) ~f
let make () =
let term = Notty_lwt.Term.create ~nosig:true ~mouse:false ~bpaste:false () in
let window = Window.make () in
let t = { term; window } in
Lwt.async (fun () ->
Lwt_stream.iter_s
(fun event ->
match Window.handle_event t.window event with
| `Done ->
let%lwt () = refresh_render t () in
Lwt.return ()
| `Terminate_program ->
raise_s [%message "Program terminated by user input"])
(Notty_lwt.Term.events t.term));
t