Class: Infopark::UserIO
- Inherits:
-
Object
- Object
- Infopark::UserIO
- Defined in:
- lib/infopark/user_io.rb,
lib/infopark/user_io/version.rb
Overview
TODO
-
beep (a) on #acknowledge, #ask or #confirm (and maybe on #listen, too)
Defined Under Namespace
Modules: Global Classes: Aborted, MissingEnv, Progress
Constant Summary collapse
- VERSION =
"0.1.1"
Class Attribute Summary collapse
-
.global ⇒ Object
Returns the value of attribute global.
Instance Method Summary collapse
- #<<(msg) ⇒ Object
- #acknowledge(*text) ⇒ Object
- #ask(*text, default: nil) ⇒ Object
- #background_other_threads ⇒ Object
- #confirm(*text) ⇒ Object
- #edit_file(kind_of_data, filename = nil) ⇒ Object
- #foreground ⇒ Object
-
#initialize(output_prefix: nil) ⇒ UserIO
constructor
A new instance of UserIO.
- #listen(prompt = nil, **options) ⇒ Object
- #new_progress(label) ⇒ Object
- #select(description, items, item_describer: :to_s, default: nil) ⇒ Object
- #start_progress(label) ⇒ Object
- #tell(*texts, newline: true, **line_options) ⇒ Object
- #tell_error(e, **options) ⇒ Object
- #tty? ⇒ Boolean
- #warn(*text) ⇒ Object
Constructor Details
#initialize(output_prefix: nil) ⇒ UserIO
Returns a new instance of UserIO.
49 50 51 52 53 54 55 56 57 58 59 |
# File 'lib/infopark/user_io.rb', line 49 def initialize(output_prefix: nil) case output_prefix when String @output_prefix = "[#{output_prefix}] " when Proc, Method @output_prefix_proc = ->() { "[#{output_prefix.call}] " } when :timestamp @output_prefix_proc = ->() { "[#{Time.now.strftime("%T.%L")}] " } end @line_pending = {} end |
Class Attribute Details
.global ⇒ Object
Returns the value of attribute global.
46 47 48 |
# File 'lib/infopark/user_io.rb', line 46 def global @global end |
Instance Method Details
#<<(msg) ⇒ Object
137 138 139 |
# File 'lib/infopark/user_io.rb', line 137 def <<(msg) tell(msg.chomp, newline: msg.end_with?("\n")) end |
#acknowledge(*text) ⇒ Object
76 77 78 79 80 81 82 |
# File 'lib/infopark/user_io.rb', line 76 def acknowledge(*text) tell("-" * 80) tell(*text, color: :cyan, bright: true) tell("-" * 80) tell("Please press ENTER to continue.") read_line end |
#ask(*text, default: nil) ⇒ Object
84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 |
# File 'lib/infopark/user_io.rb', line 84 def ask(*text, default: nil) # TODO implementation error if default not boolean or nil tell("-" * 80) tell(*text, color: :cyan, bright: true) tell("-" * 80) default_answer = default ? "yes" : "no" unless default.nil? tell("(yes/no) #{default_answer && "[#{default_answer}] "}> ", newline: false) until %w(yes no).include?(answer = read_line.strip.downcase) if answer.empty? answer = default_answer break end tell("I couldn't understand “#{answer}”.", newline: false, color: :red, bright: true) tell(" > ", newline: false) end answer == "yes" end |
#background_other_threads ⇒ Object
120 121 122 123 124 125 |
# File 'lib/infopark/user_io.rb', line 120 def background_other_threads unless @foreground_thread @background_lines = [] @foreground_thread = Thread.current end end |
#confirm(*text) ⇒ Object
108 109 110 |
# File 'lib/infopark/user_io.rb', line 108 def confirm(*text) ask(*text) or raise Aborted end |
#edit_file(kind_of_data, filename = nil) ⇒ Object
145 146 147 148 149 150 151 152 153 154 155 156 |
# File 'lib/infopark/user_io.rb', line 145 def edit_file(kind_of_data, filename = nil) wait_for_foreground if background? editor = ENV['EDITOR'] or raise MissingEnv, "No EDITOR specified." filename ||= Tempfile.new("").path tell("Start editing #{kind_of_data} using #{editor}…") sleep(1.7) system(editor, filename) File.read(filename) end |
#foreground ⇒ Object
127 128 129 130 131 132 133 134 135 |
# File 'lib/infopark/user_io.rb', line 127 def foreground if @foreground_thread @background_lines.each(&STDOUT.method(:write)) @foreground_thread = nil # take over line_pending from background @line_pending[false] = @line_pending[true] @line_pending[true] = false end end |
#listen(prompt = nil, **options) ⇒ Object
102 103 104 105 106 |
# File 'lib/infopark/user_io.rb', line 102 def listen(prompt = nil, **) prompt << " " if prompt tell("#{prompt}> ", **, newline: false) read_line.strip end |
#new_progress(label) ⇒ Object
112 113 114 |
# File 'lib/infopark/user_io.rb', line 112 def new_progress(label) Progress.new(label, self) end |
#select(description, items, item_describer: :to_s, default: nil) ⇒ Object
158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 |
# File 'lib/infopark/user_io.rb', line 158 def select(description, items, item_describer: :to_s, default: nil) return if items.empty? describer = case item_describer when Method, Proc item_describer else ->(item) { item.send(item_describer) } end choice = nil if items.size == 1 choice = items.first tell("Selected #{describer.call(choice)}.", color: :yellow) return choice end items = items.sort_by(&describer) tell("-" * 80) tell("Please select #{description}:", color: :cyan, bright: true) items.each_with_index do |item, i| tell("#{i + 1}: #{describer.call(item)}", color: :cyan, bright: true) end tell("-" * 80) default_index = items.index(default) default_selection = "[#{default_index + 1}] " if default_index until choice tell("Your choice #{default_selection}> ", newline: false) answer = read_line.strip if answer.empty? choice = default else int_answer = answer.to_i if int_answer.to_s != answer tell("Please enter a valid integer.") elsif int_answer < 1 || int_answer > items.size tell("Please enter a number from 1 through #{items.size}.") else choice = items[int_answer - 1] end end end choice end |
#start_progress(label) ⇒ Object
116 117 118 |
# File 'lib/infopark/user_io.rb', line 116 def start_progress(label) new_progress(label).tap(&:start) end |
#tell(*texts, newline: true, **line_options) ⇒ Object
61 62 63 64 65 66 |
# File 'lib/infopark/user_io.rb', line 61 def tell(*texts, newline: true, **) lines = texts.flatten.map {|text| text.to_s.split("\n", -1) }.flatten lines[0...-1].each {|line| tell_line(line, **) } tell_line(lines.last, newline: newline, **) end |
#tell_error(e, **options) ⇒ Object
72 73 74 |
# File 'lib/infopark/user_io.rb', line 72 def tell_error(e, **) tell(e, **, color: :red, bright: true) end |
#tty? ⇒ Boolean
141 142 143 |
# File 'lib/infopark/user_io.rb', line 141 def tty? STDOUT.tty? end |
#warn(*text) ⇒ Object
68 69 70 |
# File 'lib/infopark/user_io.rb', line 68 def warn(*text) tell(*text, color: :yellow, bright: true) end |