diff --git a/CHANGELOG.md b/CHANGELOG.md index b0e8e38..fe5e935 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -1,5 +1,15 @@ # Change log +## [v0.24.0] - unreleased + +### Changed +* Change #select and #multi_select to accept a :confirm_keys option, instead of always defaulting to space and return by Eleni Lixourioti(@Geekfish) +* Change #multi_select to accept a :select_keys option, instead of always defaulting to space by Eleni Lixourioti(@Geekfish) + +### Fixed +* Fix #select and #multi_select hint to print out the correct confirm_keys by Eleni Lixourioti(@Geekfish) +* Fix typo in #slider show_help by Eleni Lixourioti(@Geekfish) + ## [v0.23.1] - 2021-04-17 ### Changed diff --git a/README.md b/README.md index 8313c85..7236426 100644 --- a/README.md +++ b/README.md @@ -89,6 +89,7 @@ Or install it yourself as: * [2.6.2.5 :per_page](#2625-per_page) * [2.6.2.6 :disabled](#2626-disabled) * [2.6.2.7 :filter](#2627-filter) + * [2.6.2.8 :confirm_keys](#2628-confirm_keys) * [2.6.3 multi_select](#263-multi_select) * [2.6.3.1 :cycle](#2631-cycle) * [2.6.3.2 :enum](#2632-enum) @@ -99,6 +100,8 @@ Or install it yourself as: * [2.6.3.7 :filter](#2637-filter) * [2.6.3.8 :min](#2638-min) * [2.6.3.9 :max](#2639-max) + * [2.6.3.10 :confirm_keys](#26310-confirm_keys) + * [2.6.3.11 :select_keys](#262311-select_keys) * [2.6.4 enum_select](#264-enum_select) * [2.6.4.1 :per_page](#2641-per_page) * [2.6.4.1 :disabled](#2641-disabled) @@ -158,7 +161,7 @@ Asking question with list of options couldn't be easier using `select` like so: ```ruby prompt.select("Choose your destiny?", %w(Scorpion Kano Jax)) # => -# Choose your destiny? (Use ↑/↓ arrow keys, press Enter to select) +# Choose your destiny? (Press ↑/↓ arrow to move and Space or Enter to select) # ‣ Scorpion # Kano # Jax @@ -404,6 +407,7 @@ end ``` Available letter casing settings are: + ```ruby :up # change to upper case :down # change to small case @@ -636,7 +640,7 @@ By default the choice name is also the value the prompt will return when selecte choices = {small: 1, medium: 2, large: 3} prompt.select("What size?", choices) # => -# What size? (Press ↑/↓ arrow to move and Enter to select) +# What size? (Press ↑/↓ arrow to move and Space or Enter to select) # ‣ small # medium # large @@ -663,7 +667,7 @@ prompt.select("What size?") do |menu| menu.choice name: "large", value: 3 end # => -# What size? (Press ↑/↓ arrow to move and Enter to select) +# What size? (Press ↑/↓ arrow to move and Space or Enter to select) # ‣ small # ✘ medium (out of stock) # large @@ -698,7 +702,7 @@ For asking questions involving list of options use `select` method by passing th ```ruby prompt.select("Choose your destiny?", %w(Scorpion Kano Jax)) # => -# Choose your destiny? (Use ↑/↓ arrow keys, press Enter to select) +# Choose your destiny? (Press ↑/↓ arrow to move and Space or Enter to select) # ‣ Scorpion # Kano # Jax @@ -713,7 +717,7 @@ prompt.select("Choose your destiny?") do |menu| menu.choice "Jax" end # => -# Choose your destiny? (Use ↑/↓ arrow keys, press Enter to select) +# Choose your destiny? (Press ↑/↓ arrow to move and Space or Enter to select) # ‣ Scorpion # Kano # Jax @@ -728,7 +732,7 @@ prompt.select("Choose your destiny?") do |menu| menu.choice "Jax", -> { "Nice choice captain!" } end # => -# Choose your destiny? (Use ↑/↓ arrow keys, press Enter to select) +# Choose your destiny? (Press ↑/↓ arrow to move and Space or Enter to select) # ‣ Scorpion # Kano # Jax @@ -753,7 +757,7 @@ prompt.select("Choose your destiny?") do |menu| menu.choice "Jax", 3 end # => -# Choose your destiny? (Use ↑/↓ arrow keys, press Enter to select) +# Choose your destiny? (Press ↑/↓ arrow to move and Space or Enter to select) # Scorpion # Kano # ‣ Jax @@ -766,7 +770,7 @@ You can navigate the choices using the arrow keys or define your own key mapping ```ruby prompt.select("Choose your destiny?", %w(Scorpion Kano Jax), cycle: true) # => -# Choose your destiny? (Use ↑/↓ arrow keys, press Enter to select) +# Choose your destiny? (Press ↑/↓ arrow to move and Space or Enter to select) # ‣ Scorpion # Kano # Jax @@ -785,7 +789,7 @@ prompt.select("Choose your destiny?") do |menu| menu.choice "Jax", 3 end # => -# Choose your destiny? (Use ↑/↓ arrow or number (0-9) keys, press Enter to select) +# Choose your destiny? (Press ↑/↓ arrow or 1-9 number to move and Space or Enter to select) # 1. Scorpion # 2. Kano # ‣ 3. Jax @@ -813,7 +817,7 @@ You can configure active marker like so: choices = %w(Scorpion Kano Jax) prompt.select("Choose your destiny?", choices, symbols: { marker: ">" }) # => -# Choose your destiny? (Use ↑/↓ and ←/→ arrow keys, press Enter to select) +# Choose your destiny? (Press ↑/↓ to move and Space or Enter to select) # > Scorpion # Kano # Jax @@ -827,7 +831,7 @@ By default the menu is paginated if selection grows beyond `6` items. To change letters = ("A".."Z").to_a prompt.select("Choose your letter?", letters, per_page: 4) # => -# Which letter? (Use ↑/↓ and ←/→ arrow keys, press Enter to select) +# Which letter? (Press ↑/↓/←/→ arrow to move and press Space or Enter to select) # ‣ A # B # C @@ -835,6 +839,7 @@ prompt.select("Choose your letter?", letters, per_page: 4) ``` You can also customise page navigation text using `:help` option: + ```ruby letters = ("A".."Z").to_a prompt.select("Choose your letter?") do |menu| @@ -870,7 +875,7 @@ The disabled choice will be displayed with a cross `✘` character next to it an ```ruby prompt.select("Choose your destiny?", warriors) # => -# Choose your destiny? (Use ↑/↓ arrow keys, press Enter to select) +# Choose your destiny? (Press ↑/↓ arrow to move and Space or Enter to select) # ‣ Scorpion # Kano # ✘ Goro (injury) @@ -887,7 +892,7 @@ To activate dynamic list searching on letter/number key presses use `:filter` op warriors = %w(Scorpion Kano Jax Kitana Raiden) prompt.select("Choose your destiny?", warriors, filter: true) # => -# Choose your destiny? (Use ↑/↓ arrow keys, press Enter to select, and letter keys to filter) +# Choose your destiny? (Press ↑/↓ arrow to move, Space or Enter to select and letters to filter) # ‣ Scorpion # Kano # Jax @@ -916,6 +921,78 @@ Filter characters can be deleted partially or entirely via, respectively, Backsp If the user changes or deletes a filter, the choices previously selected remain selected. +#### 2.6.2.8 `:confirm_keys` + +You can configure which key(s) confirm your selection (`:space` and `:return` by default): + +```ruby +choices = %w(Scorpion Kano Jax) +prompt.select("Choose your destiny?", choices, confirm_keys: [:ctrl_s, ","]) +# => +# Choose your destiny? (Press ↑/↓ to move and Ctrl+S or , to select) +# ‣ Scorpion +# Kano +# Jax +``` + +This is particularly useful in conjunction with `:filter`, as you may have choices that include spaces. + +```ruby +choices = ["Jax Sr", "Jax", "Jax Jr"] +prompt.select("Choose your destiny?", choices, confirm_keys: [:ctrl_s], filter: true) +# => +# Choose your destiny? (Press ↑/↓ to move, Ctrl+S to select and letters to filter) +# ‣ Jax Sr +# Jax +# Jax Jr +``` + +After the user types "Jax": + +```ruby +# => +# Choose your destiny? (Filter: "Jax") +# ‣ Jax Sr +# Jax +# Jax Jr +``` + +After the user presses space: + +```ruby +# => +# Choose your destiny? (Filter: "Jax ") +# ‣ Jax Sr +# Jax Jr +``` + +You may also pass your own key labels to be displayed in the hint: + +```ruby +choices = ["Jax Sr", "Jax", "Jax Jr"] +prompt.select("Choose your destiny?", choices, confirm_keys: [:enter, {ctrl_s: "Ctrl-S"}, {"," => "Comma (,)"}]) +# => +# Choose your destiny? (Press ↑/↓ arrow to move and Enter, Ctrl-S or Comma (,) to select) +# ‣ Jax Sr +# Jax +# Jax Jr +``` + +Confirm keys may be configured using the DSL, similar to `:choices`: + +```ruby +choices = ["Jax Sr", "Jax", "Jax Jr"] +prompt.select("Choose your destiny?") do |menu| + menu.choices choices + menu.confirm_keys :enter, {ctrl_s: "Ctrl-S"}, {"," => "Comma (,)"} +end +# => +# Choose your destiny? (Press ↑/↓ arrow to move and Enter, Ctrl-S or Comma (,) to select) +# ‣ Jax Sr +# Jax +# Jax Jr +``` + ### 2.6.3 multi_select For asking questions involving multiple selection list use `multi_select` method by passing the question and possible choices: @@ -925,7 +1002,7 @@ choices = %w(vodka beer wine whisky bourbon) prompt.multi_select("Select drinks?", choices) # => # -# Select drinks? (Use ↑/↓ arrow keys, press Space to select and Enter to finish)" +# Select drinks? (Press ↑/↓ arrow keys to move, Space/Ctrl+A|R to select (all|rev) and Enter to finish) # ‣ ⬡ vodka # ⬡ beer # ⬡ wine @@ -1038,7 +1115,7 @@ By default the menu is paginated if selection grows beyond `6` items. To change letters = ("A".."Z").to_a prompt.multi_select("Choose your letter?", letters, per_page: 4) # => -# Which letter? (Use ↑/↓ and ←/→ arrow keys, press Space to select and Enter to finish) +# Which letter? (Press ↑/↓/←/→ arrow keys to move, Space/Ctrl+A|R to select (all|rev) and Enter to finish) # ‣ ⬡ A # ⬡ B # ⬡ C @@ -1065,7 +1142,7 @@ The disabled choice will be displayed with a cross `✘` character next to it an ```ruby prompt.multi_select("Choose your favourite drink?", drinks) # => -# Choose your favourite drink? (Use ↑/↓ arrow keys, press Space to select and Enter to finish) +# Choose your favourite drink? (Press ↑/↓ arrow keys to move, Space/Ctrl+A|R to select (all|rev) and Enter to finish) # ‣ ⬡ bourbon # ✘ sake (out of stock) # ⬡ vodka @@ -1083,7 +1160,7 @@ header use the :echo option: choices = %w(vodka beer wine whisky bourbon) prompt.multi_select("Select drinks?", choices, echo: false) # => -# Select drinks? +# Select drinks? (Press ↑/↓ arrow keys to move, Space/Ctrl+A|R to select (all|rev) and Enter to finish) # ⬡ vodka # ⬢ 2) beer # ⬡ 3) wine @@ -1099,7 +1176,7 @@ To activate dynamic list filtering on letter/number typing, use the :filter opti choices = %w(vodka beer wine whisky bourbon) prompt.multi_select("Select drinks?", choices, filter: true) # => -# Select drinks? (Use ↑/↓ arrow keys, press Space to select and Enter to finish, and letter keys to filter) +# Select drinks? (Press ↑/↓ arrow keys to move, Space/Ctrl+A|R to select (all|rev), Enter to finish and letters to filter) # ‣ ⬡ vodka # ⬡ beer # ⬡ wine @@ -1153,6 +1230,105 @@ prompt.multi_select("Select drinks?", choices, max: 3) # ‣ ⬡ bourbon ``` +#### 2.6.3.10 `:confirm_keys` + +This works similar to the [same option for `select`](#2628-confirm_keys). +You can configure which key(s) confirm your selection (`:return` by default): + +```ruby +choices = %w(vodka beer wine whisky bourbon) +prompt.multi_select("Select drinks?", choices, confirm_keys: [:ctrl_s, ","]) +# => +# Select drinks? vodka, beer, whisky +# ⬢ vodka +# ⬢ beer +# ⬡ wine +# ⬢ whisky +# ‣ ⬡ bourbon +``` + +The user can then press the `Ctrl+S` or the `,` key to confirm their selection: + +```ruby +# => +# Select drinks? vodka, beer, whisky +``` + +You may also configure the keys using the DSL, in the same way that is done for [select](#2628-confirm_keys). + +#### 2.6.3.11 `:select_keys` + +You can configure which key selects an option (`:space` default). +This is particularly useful in conjunction with the `filter` option, as you may have choices that include spaces. + +```ruby +choices = ["gin", "gin tonic", "gin fizz", "beer"] +prompt.multi_select("Select drinks?", choices, filter: true, select_keys: [:ctrl_s]) +# => +# Select drinks? (Press ↑/↓ arrow keys to move, Ctrl+S/Ctrl+A|R to select (all|rev), Enter to finish and letters to filter) +# ‣ ⬡ gin +# ⬡ gin tonic +# ⬡ gin fizz +# ⬡ beer +``` + +After the user types "gin": + +```ruby +# => +# Select drinks? (Filter: "gin") +# ‣ ⬡ gin +# ⬡ gin tonic +# ⬡ gin fizz +``` + +and then presses space: + +```ruby +# => +# Select drinks? (Filter: "gin ") +# ‣ ⬡ gin tonic +# ⬡ gin fizz +``` + +The user can then press the `Ctrl+S` key combo to select the options: + +```ruby +# => +# Select drinks? (Filter: "gin ") +# ‣ ⬢ gin tonic +# ⬡ gin fizz +``` + +Similar to the `:confirm_keys` option, you may also pass your own key labels to be displayed in the hint: + +```ruby +choices = ["gin", "gin tonic", "gin fizz", "beer"] +prompt.multi_select("Select drinks?", choices, filter: true, select_keys: [{ctrl_s: "Ctrl-S"}, {space: "Spacebar"}]) +# => +# Select drinks? (Press ↑/↓ arrow keys to move, Ctrl-S or Spacebar/Ctrl+A|R to select (all|rev), Enter to finish and letters to filter) +# ‣ ⬡ gin +# ⬡ gin tonic +# ⬡ gin fizz +# ⬡ beer +``` + +You can also configure your keys using the DSL, similar to `:confirm_keys`: + +```ruby +choices = ["gin", "gin tonic", "gin fizz", "beer"] +prompt.multi_select("Select drinks?") do |menu| + menu.choices choices + menu.select_keys({ctrl_s: "Ctrl-S"}) +end +# => +# Select drinks? (Press ↑/↓ arrow keys to move, Ctrl-S or Spacebar/Ctrl+A|R to select (all|rev), and Enter to finish) +# ‣ ⬡ gin +# ⬡ gin tonic +# ⬡ gin fizz +# ⬡ beer +``` + ### 2.6.4 enum_select In order to ask for standard selection from indexed list you can use `enum_select` and pass question together with possible choices: @@ -1655,7 +1831,7 @@ You could also use `pastel`: ```ruby notice = Pastel.new.cyan.on_blue.detach prompt = TTY::Prompt.new(active_color: notice) -```` +``` Or use coloring of your own choice: @@ -1698,7 +1874,7 @@ You could also use `pastel`: ```ruby notice = Pastel.new.cyan.on_blue.detach prompt = TTY::Prompt.new(help_color: notice) -```` +``` Or use coloring of your own choice: @@ -1746,7 +1922,7 @@ Prompts such as `select`, `multi_select`, `expand`, `slider` support `:quiet` wh prompt = TTY::Prompt.new(quiet: true) # single prompt prompt.select("What is your favorite color?", %w(blue yellow orange)) -```` +``` ### 3.8 `:track_history` diff --git a/examples/multi_select_custom_keys.rb b/examples/multi_select_custom_keys.rb new file mode 100644 index 0000000..35a95ec --- /dev/null +++ b/examples/multi_select_custom_keys.rb @@ -0,0 +1,10 @@ +# frozen_string_literal: true + +require_relative "../lib/tty-prompt" + +prompt = TTY::Prompt.new + +drinks = %w[vodka beer wine whisky bourbon] +prompt.multi_select("Choose your favourite drink?", drinks, + confirm_keys: [:return, {escape: "Esc"}, "."], + select_keys: [{space: "Spacebar"}, :ctrl_s, ","]) diff --git a/examples/multi_select_custom_keys_dsl.rb b/examples/multi_select_custom_keys_dsl.rb new file mode 100644 index 0000000..760e3eb --- /dev/null +++ b/examples/multi_select_custom_keys_dsl.rb @@ -0,0 +1,12 @@ +# frozen_string_literal: true + +require_relative "../lib/tty-prompt" + +prompt = TTY::Prompt.new + +drinks = %w[vodka beer wine whisky bourbon] +prompt.multi_select("Choose your favourite drink?") do |menu| + menu.choices drinks + menu.confirm_keys :return, {escape: "Esc"}, "." + menu.select_keys({space: "Spacebar"}, :ctrl_s, ",") +end diff --git a/examples/multi_select_filtered.rb b/examples/multi_select_filtered.rb new file mode 100644 index 0000000..be7f5fa --- /dev/null +++ b/examples/multi_select_filtered.rb @@ -0,0 +1,8 @@ +# frozen_string_literal: true + +require_relative "../lib/tty-prompt" + +prompt = TTY::Prompt.new + +drinks = %w[vodka beer wine whisky bourbon] +prompt.multi_select("Choose your favourite drink?", drinks, filter: true) diff --git a/examples/select_confirm_key_labels.rb b/examples/select_confirm_key_labels.rb new file mode 100644 index 0000000..05320f4 --- /dev/null +++ b/examples/select_confirm_key_labels.rb @@ -0,0 +1,12 @@ +# frozen_string_literal: true + +require_relative "../lib/tty-prompt" + +prompt = TTY::Prompt.new + +warriors = %w[Scorpion Kano Jax Kitana Raiden] + +answer = prompt.select("Choose your destiny?", warriors, + confirm_keys: [:enter, {ctrl_s: "Ctrl-S"}, ","]) + +puts answer.inspect \ No newline at end of file diff --git a/examples/select_filtered_with_spaces.rb b/examples/select_filtered_with_spaces.rb new file mode 100644 index 0000000..92325c0 --- /dev/null +++ b/examples/select_filtered_with_spaces.rb @@ -0,0 +1,12 @@ +# frozen_string_literal: true + +require_relative "../lib/tty-prompt" + +prompt = TTY::Prompt.new + +warriors = ["Jax", "Jax Jr", "Kitana", "Raiden ft. Thunder"] + +answer = prompt.select("Choose your destiny?", warriors, + filter: true, confirm_keys: %i[return ctrl_s]) + +puts answer.inspect diff --git a/lib/tty/prompt/list.rb b/lib/tty/prompt/list.rb index 3ad585a..0d53bfd 100644 --- a/lib/tty/prompt/list.rb +++ b/lib/tty/prompt/list.rb @@ -14,14 +14,23 @@ class Prompt # @api private class List # Allowed keys for filter, along with backspace and canc. - FILTER_KEYS_MATCHER = /\A([[:alnum:]]|[[:punct:]])\Z/.freeze + FILTER_KEYS_MATCHER = /\A([[:alnum:]]|[[:punct:]]|[[:blank:]])\Z/.freeze # Checks type of default parameter to be integer INTEGER_MATCHER = /\A\d+\Z/.freeze + # The default keys that confirm the selected item(s) + DEFAULT_CONFIRM_KEYS = %i[space return enter].freeze + + # The keys that signify "end of line" (EOL). + # Depending on whether we are on a Unix system / Windows + # the "Enter" key may translate to CR and/or LF characters. + # See also List#ensure_eol_compat + EOL_KEYS = %i[enter return].freeze + # Create instance of TTY::Prompt::List menu. # - # @param Hash options + # @param [Hash] options # the configuration options # @option options [Symbol] :default # the default active choice, defaults to 1 @@ -31,6 +40,8 @@ class List # the marker for the selected item # @option options [String] :enum # the delimiter for the item index + # @option options [Array String}>] + # :confirm_keys the key(s) to confirm the selected item(s) # # @api public def initialize(prompt, **options) @@ -47,6 +58,9 @@ def initialize(prompt, **options) @filterable = options.fetch(:filter) { false } @symbols = @prompt.symbols.merge(options.fetch(:symbols, {})) @quiet = options.fetch(:quiet) { @prompt.quiet } + @confirm_keys = init_action_keys(options.fetch(:confirm_keys) do + self.class::DEFAULT_CONFIRM_KEYS + end) @filter = [] @filter_cache = {} @help = options[:help] @@ -79,6 +93,105 @@ def default(*default_values) @default = default_values end + # Set confirm keys + # + # @param [Array String}>] keys + # the key(s) to confirm the selected item(s) + # + # @return [Hash{Symbol, String => String}] + # + # @api public + def confirm_keys(*keys) + keys = keys.flatten + return @confirm_keys if keys.empty? + + @confirm_keys = init_action_keys(keys) + end + + # Initialize any default or custom action keys + # setting up their labels and dealing with compat + # + # @param [Array String}>] keys + # the key(s) as only name or name and label pair + # + # @return [Hash{Symbol, String => String}] + # + # @api private + def init_action_keys(keys) + keys = keys_with_labels(keys) + ensure_eol_compat(keys) + end + + # Normalize a list of key symbols or symbol-label hashes + # into a single symbol-label lookup hash. + # + # @example Only with symbol keys + # keys = [:enter, :ctrl_s] + # keys_with_labels(keys) + # # => {enter: "Enter", ctrl_s: "Ctrl+S"} + # + # @example With mixed keys + # keys = [:enter, {ctrl_s: "Ctrl-S"}] + # keys_with_labels(keys) + # # => {enter: "Enter", ctrl_s: "Ctrl-S"} + # + # @param [Array String}>] keys + # the key(s) as only name or name and label pair + # + # @return [Hash{Symbol, String => String}] + # + # @api private + def keys_with_labels(keys) + keys.reduce({}) do |result, key| + obj = key.is_a?(::Hash) ? key : {key => key_help_label(key)} + result.merge(obj) + end + end + + # Convert a key name into a human-readable label + # + # @param [Symbol, String] key_name + # the key name to convert to label + # + # @return [String] + # + # @api private + def key_help_label(key_name) + if key_name == :return + "Enter" + else + key_name.to_s.split("_").map(&:capitalize).join("+") + end + end + + # Ensure that if any EOL char is passed as an action key + # then all EOL chars are included (for cross-system compat) + # Maintain any custom labels. + # + # @example + # keys = {return: "Enter", ctrl_s: "Ctrl+S"} + # ensure_eol_compat(keys) + # # => {enter: "Enter", return: "Enter", ctrl_s: "Ctrl+S"} + # + # @param [Hash{Symbol, String => String}] keys + # the key(s) as name and label pair + # + # @return [Hash{Symbol, String => String}] + # + # @api private + def ensure_eol_compat(keys) + key_symbols = keys.keys.sort_by(&:to_s) + key_intersection = EOL_KEYS & key_symbols + + if key_intersection.empty? || key_intersection == EOL_KEYS + keys + else + eol_label = keys[key_intersection.first] + missing_key = (EOL_KEYS - key_intersection).first + keys.merge({missing_key => eol_label}) + end + end + # Select paginator based on the current navigation key # # @return [Paginator] @@ -143,7 +256,7 @@ def help(value = (not_set = true)) # # @api public def show_help(value = (not_set = true)) - return @show_ehlp if not_set + return @show_help if not_set @show_help = value end @@ -163,11 +276,38 @@ def arrows_help arrows.join end + # Information about keys that confirm the selection + # + # @example Get help string for many keys + # keys = {return: "Enter", ctrl_s: "Ctrl+S", space: "Space"} + # keys_help(keys) + # # => "Enter, Ctrl+S or Space" + # + # @example Get help string for one key + # keys = {return: "Enter"} + # keys_help(keys) + # # => "Enter" + # + # @param [Hash{Symbol, String => String}] keys + # the key(s) as name and label pair + # + # @return [String] + # + # @api private + def keys_help(keys) + labels = keys.values.uniq + if labels.length == 1 + labels[0] + else + "#{labels[0..-2].join(', ')} or #{labels[-1]}" + end + end + # Default help text # # Note that enumeration and filter are mutually exclusive # - # @a public + # @api public def default_help str = [] str << "(Press " @@ -175,7 +315,7 @@ def default_help str << " or 1-#{choices.size} number" if enumerate? str << " to move" str << (filterable? ? "," : " and") - str << " Enter to select" + str << " #{keys_help(@confirm_keys)} to select" str << " and letters to filter" if filterable? str << ")" str.join @@ -233,8 +373,8 @@ def choices(values = (not_set = true)) # Call the list menu by passing question and choices # # @param [String] question + # @param [Array[Object]] possibilities # - # @param # @api public def call(question, possibilities, &block) choices(possibilities) @@ -263,12 +403,6 @@ def keynum(event) @active = value end - def keyenter(*) - @done = true unless choices.empty? - end - alias keyreturn keyenter - alias keyspace keyenter - def search_choice_in(searchable) searchable.find { |i| !choices[i - 1].disabled? } end @@ -305,7 +439,6 @@ def keydown(*) @paging_changed = @by_page @by_page = false end - alias keytab keydown # Moves all choices page by page keeping the current selected item # at the same level on each page. @@ -348,10 +481,20 @@ def keyleft(*) end alias keypage_up keyleft - def keypress(event) - return unless filterable? + # Callback fired when a confirm key is pressed + # + # @api private + def confirm + @done = true unless choices.empty? + end - if event.value =~ FILTER_KEYS_MATCHER + def keypress(event) + if @confirm_keys.keys.include?(event.key.name) || + @confirm_keys.keys.include?(event.value) + confirm + elsif event.key.name == :tab + keydown + elsif filterable? && event.value =~ FILTER_KEYS_MATCHER @filter << event.value @active = 1 end @@ -487,7 +630,7 @@ def answer # Clear screen lines # - # @param [String] + # @param [Integer] lines # # @api private def refresh(lines) diff --git a/lib/tty/prompt/multi_list.rb b/lib/tty/prompt/multi_list.rb index 0c50043..d1abf84 100644 --- a/lib/tty/prompt/multi_list.rb +++ b/lib/tty/prompt/multi_list.rb @@ -10,15 +10,26 @@ class Prompt # # @api private class MultiList < List + # The default keys that confirm the selected item(s) + DEFAULT_CONFIRM_KEYS = %i[return enter].freeze + + # The default keys that select choices + DEFAULT_SELECT_KEYS = %i[space].freeze + # Create instance of TTY::Prompt::MultiList menu. # - # @param [Prompt] :prompt + # @param [Prompt] prompt # @param [Hash] options + # @option options [Array String}>] + # :select_keys the key(s) used for selecting choices # # @api public def initialize(prompt, **options) super @selected = SelectedChoices.new + @select_keys = init_select_keys(options.fetch(:select_keys) do + self.class::DEFAULT_SELECT_KEYS + end) @help = options[:help] @echo = options.fetch(:echo, true) @min = options[:min] @@ -39,22 +50,45 @@ def max(value) @max = value end - # Callback fired when enter/return key is pressed + # Callback fired when a confirm key is pressed # # @api private - def keyenter(*) + def confirm valid = true valid = @min <= @selected.size if @min valid = @selected.size <= @max if @max super if valid end - alias keyreturn keyenter - # Callback fired when space key is pressed + # @see List#confirm_keys + # + # @api public + def confirm_keys(*keys) + super + check_conflicting_keys + @confirm_keys + end + + # Set select keys + # + # @param [Array String}>] keys + # the key(s) used for selecting choices + # + # @return [Hash{Symbol, String => String}] + # + # @api public + def select_keys(*keys) + keys = keys.flatten + return @select_keys if keys.empty? + + @select_keys = init_select_keys(keys) + end + + # Callback fired when the selection key is pressed # # @api private - def keyspace(*) + def select_choice active_choice = choices[@active - 1] if @selected.include?(active_choice) @selected.delete_at(@active - 1) @@ -65,6 +99,18 @@ def keyspace(*) end end + # Callback fired when any key is pressed + # + # @api private + def keypress(event) + if @select_keys.keys.include?(event.key.name) || + @select_keys.keys.include?(event.value) + select_choice + else + super(event) + end + end + # Selects all choices when Ctrl+A is pressed # # @api private @@ -89,6 +135,33 @@ def keyctrl_r(*) private + # Initialize any default or custom select keys + # setting up their labels and dealing with any key conflicts + # + # @see List#init_action_keys + # + # @api private + def init_select_keys(keys) + @select_keys = init_action_keys(keys) + check_conflicting_keys + @select_keys + end + + # Checks that there are no key options clashing + # + # @raise [ConfigurationError] + # + # @api private + def check_conflicting_keys + conflicting_keys = @confirm_keys.keys & @select_keys.keys + return if conflicting_keys.empty? + + raise ConfigurationError, + ":confirm_keys and :select_keys cannot use the same " \ + "#{conflicting_keys.map(&:inspect).join(', ')} " \ + "key#{'s' if conflicting_keys.size > 1}" + end + # Setup default options and active selection # # @api private @@ -146,12 +219,12 @@ def default_help str << "(Press " str << "#{arrows_help} arrow" str << " or 1-#{choices.size} number" if enumerate? - str << " to move, Space" + str << " to move, #{keys_help(@select_keys)}" str << "/Ctrl+A|R" if @max.nil? str << " to select" str << " (all|rev)" if @max.nil? str << (filterable? ? "," : " and") - str << " Enter to finish" + str << " #{keys_help(@confirm_keys)} to finish" str << " and letters to filter" if filterable? str << ")" str.join diff --git a/lib/tty/prompt/slider.rb b/lib/tty/prompt/slider.rb index 68453c9..6f40a94 100644 --- a/lib/tty/prompt/slider.rb +++ b/lib/tty/prompt/slider.rb @@ -95,7 +95,7 @@ def help(text = (not_set = true)) # # @api public def show_help(value = (not_set = true)) - return @show_ehlp if not_set + return @show_help if not_set @show_help = value end diff --git a/spec/unit/multi_select_spec.rb b/spec/unit/multi_select_spec.rb index bf09b61..15861c2 100644 --- a/spec/unit/multi_select_spec.rb +++ b/spec/unit/multi_select_spec.rb @@ -84,6 +84,36 @@ def exit_message(prompt, choices) expect(prompt.output.string).to eq(expected_output) end + it "selects item when custom key pressed and shows custom key labels" do + choices = %w[vodka beer wine whisky bourbon] + prompt.input << "\C-s\r" + prompt.input.rewind + expect(prompt.multi_select("Select drinks?", choices, select_keys: [:ctrl_s, {escape: "Esc"}])).to eq(["vodka"]) + + expected_output = + output_helper("Select drinks?", choices, "vodka", [], init: true, + hint: "Press #{up_down} arrow to move, Ctrl+S or Esc/Ctrl+A|R to select (all|rev) and Enter to finish") + + output_helper("Select drinks?", choices, "vodka", ["vodka"]) + + exit_message("Select drinks?", %w[vodka]) + + expect(prompt.output.string).to eq(expected_output) + end + + it "selects item and confirms selection with custom keys" do + choices = %w[vodka beer wine whisky bourbon] + prompt.input << "\C-s\e" + prompt.input.rewind + expect(prompt.multi_select("Select drinks?", choices, select_keys: [:ctrl_s], confirm_keys: [{escape: "Esc"}])).to eq(["vodka"]) + + expected_output = + output_helper("Select drinks?", choices, "vodka", [], init: true, + hint: "Press #{up_down} arrow to move, Ctrl+S/Ctrl+A|R to select (all|rev) and Esc to finish") + + output_helper("Select drinks?", choices, "vodka", ["vodka"]) + + exit_message("Select drinks?", %w[vodka]) + + expect(prompt.output.string).to eq(expected_output) + end + it "selects item when space pressed but doesn't echo item if echo: false" do choices = %w[vodka beer wine whisky bourbon] prompt.input << " \r" @@ -210,6 +240,29 @@ def exit_message(prompt, choices) expect(prompt.output.string).to eq(expected_output) end + it "sets confirm_keys and select_keys through DSL" do + choices = %w[vodka beer wine] + prompt.on(:keypress) { |e| prompt.trigger(:keydown) if e.value == "j" } + prompt.input << "j\r\C-s" + prompt.input.rewind + value = prompt.multi_select("Select drinks?") do |menu| + menu.choices choices + menu.confirm_keys :ctrl_s + menu.select_keys :enter + end + expect(value).to eq(["beer"]) + + expected_output = + output_helper("Select drinks?", choices, "vodka", [], init: true, + hint: "Press #{up_down} arrow to move, Enter/Ctrl+A|R to select " \ + "(all|rev) and Ctrl+S to finish") + + output_helper("Select drinks?", choices, "beer", []) + + output_helper("Select drinks?", choices, "beer", %w[beer]) + + exit_message("Select drinks?", %w[beer]) + + expect(prompt.output.string).to eq(expected_output) + end + it "sets default options through hash syntax" do prompt.input << "\r" prompt.input.rewind @@ -251,6 +304,24 @@ def exit_message(prompt, choices) /default index `6` out of range \(1 - 5\)/) end + it "raises error when confirm and select keys clash (with default select_key)" do + prompt.input << "\r" + prompt.input.rewind + expect { + prompt.multi_select("Select drinks?", %w[vodka beer wine], confirm_keys: [:space]) + }.to raise_error(TTY::Prompt::ConfigurationError, + ":confirm_keys and :select_keys cannot use the same :space key") + end + + it "raises error when confirm and select keys clash (configured)" do + prompt.input << "\r" + prompt.input.rewind + expect { + prompt.multi_select("Select drinks?", %w[vodka beer wine], confirm_keys: %i[space ctrl_s], select_keys: [:space]) + }.to raise_error(TTY::Prompt::ConfigurationError, + ":confirm_keys and :select_keys cannot use the same :space key") + end + it "sets prompt prefix" do prompt = TTY::Prompt::Test.new(prefix: "[?] ") choices = %w[vodka beer wine whisky bourbon] @@ -709,6 +780,37 @@ def exit_message(prompt, choices) expect(prompt.output.string).to eql(expected_output) end + + it "continues filtering when space is pressed with custom select key" do + choices = ["gin", "gin fizz", "gin tonic"] + prompt.input << "gin" + prompt.input << " " + prompt.input << "f" + prompt.input << "\C-s" + prompt.input << "\u007F" # Delete one letter + prompt.input << "t" + prompt.input << "\C-s" + prompt.input << "\r" + prompt.input.rewind + + expect(prompt.multi_select("Select drinks?", choices, filter: true, select_keys: [:ctrl_s])).to eq(["gin fizz", "gin tonic"]) + + expected_output = + output_helper("Select drinks?", choices, "gin", [], init: true, + hint: "Press #{up_down} arrow to move, Ctrl+S/Ctrl+A|R to select (all|rev), Enter to finish and letters to filter") + + output_helper("Select drinks?", choices, "gin", [], hint: "Filter: \"g\"") + + output_helper("Select drinks?", choices, "gin", [], hint: "Filter: \"gi\"") + + output_helper("Select drinks?", choices, "gin", [], hint: "Filter: \"gin\"") + + output_helper("Select drinks?", ["gin fizz", "gin tonic"], "gin fizz", [], hint: "Filter: \"gin \"") + + output_helper("Select drinks?", ["gin fizz"], "gin fizz", [], hint: "Filter: \"gin f\"") + + output_helper("Select drinks?", ["gin fizz"], "gin fizz", ["gin fizz"], hint: "Filter: \"gin f\"") + + output_helper("Select drinks?", ["gin fizz", "gin tonic"], "gin fizz", ["gin fizz"], hint: "Filter: \"gin \"") + + output_helper("Select drinks?", ["gin tonic"], "gin tonic", ["gin fizz"], hint: "Filter: \"gin t\"") + + output_helper("Select drinks?", ["gin tonic"], "gin tonic", ["gin fizz", "gin tonic"], hint: "Filter: \"gin t\"") + + exit_message("Select drinks?", ["gin fizz", "gin tonic"]) + + expect(prompt.output.string).to eq(expected_output) + end end context "with :disabled" do diff --git a/spec/unit/select_spec.rb b/spec/unit/select_spec.rb index 475909f..da089e5 100644 --- a/spec/unit/select_spec.rb +++ b/spec/unit/select_spec.rb @@ -53,7 +53,7 @@ def exit_message(prompt, choice) expected_output = output_helper("What size?", choices, :Large, init: true, - hint: "Press #{up_down} arrow to move and Enter to select") + + hint: "Press #{up_down} arrow to move and Space or Enter to select") + exit_message("What size?", "Large") expect(prompt.output.string).to eq(expected_output) @@ -68,7 +68,7 @@ def exit_message(prompt, choice) end expect { prompt.select("What size?", choices) }.not_to output.to_stderr expect(prompt.output.string).to eq([ - "\e[?25lWhat size? \e[90m(Press #{up_down} arrow to move and Enter to select)\e[0m\n", + "\e[?25lWhat size? \e[90m(Press #{up_down} arrow to move and Space or Enter to select)\e[0m\n", "\e[32m#{symbols[:marker]} Large\e[0m\n", " Medium\n", " Small", @@ -90,7 +90,7 @@ def exit_message(prompt, choice) prompt.input.rewind expect(prompt.select("What size?", choices, default: 1)).to eq(1) expect(prompt.output.string).to eq([ - "\e[?25lWhat size? \e[90m(Press #{up_down} arrow to move and Enter to select)\e[0m\n", + "\e[?25lWhat size? \e[90m(Press #{up_down} arrow to move and Space or Enter to select)\e[0m\n", "\e[32m#{symbols[:marker]} large\e[0m\n", " medium\n", " small", @@ -107,7 +107,7 @@ def exit_message(prompt, choice) expect(prompt.select("What size?", choices, default: 1)).to eq(nil) expect(prompt.output.string).to eq([ output_helper("What size?", choices.keys, :none, init: true, - hint: "Press #{up_down} arrow to move and Enter to select"), + hint: "Press #{up_down} arrow to move and Space or Enter to select"), exit_message("What size?", "none") ].join) end @@ -124,7 +124,7 @@ def exit_message(prompt, choice) end expect(value).to eq("Large") expect(prompt.output.string).to eq([ - "\e[?25lWhat size? \e[90m(Press #{up_down} arrow to move and Enter to select)\e[0m\n", + "\e[?25lWhat size? \e[90m(Press #{up_down} arrow to move and Space or Enter to select)\e[0m\n", "\e[32m> Large\e[0m\n", " Medium\n", " Small", @@ -134,6 +134,22 @@ def exit_message(prompt, choice) ].join) end + it "sets confirm_keys through DSL" do + choices = %w[Large Medium Small] + prompt.input << "\C-s" + prompt.input.rewind + value = prompt.select("What size?") do |menu| + menu.choices choices + menu.confirm_keys :ctrl_s + end + expect(value).to eq("Large") + expect(prompt.output.string).to eq([ + output_helper("What size?", choices, "Large", init: true, + hint: "Press #{up_down} arrow to move and Ctrl+S to select"), + exit_message("What size?", "Large") + ].join) + end + it "sets choice name & value through DSL" do prompt = TTY::Prompt::Test.new(symbols: {marker: ">"}) prompt.input << " " @@ -145,7 +161,7 @@ def exit_message(prompt, choice) end expect(value).to eq(1) expect(prompt.output.string).to eq([ - "\e[?25lWhat size? \e[90m(Press #{up_down} arrow to move and Enter to select)\e[0m\n", + "\e[?25lWhat size? \e[90m(Press #{up_down} arrow to move and Space or Enter to select)\e[0m\n", "\e[32m> large\e[0m\n", " medium\n", " small", @@ -164,7 +180,7 @@ def exit_message(prompt, choice) end expect(value).to eq("Large") expect(prompt.output.string).to eq([ - "\e[?25lWhat size? \e[90m(Press #{up_down} arrow to move and Enter to select)\e[0m\n", + "\e[?25lWhat size? \e[90m(Press #{up_down} arrow to move and Space or Enter to select)\e[0m\n", "\e[32m#{symbols[:marker]} Large\e[0m\n", " Medium\n", " Small", @@ -187,7 +203,7 @@ def exit_message(prompt, choice) end expect(value).to eq(2) expect(prompt.output.string).to eq([ - "\e[?25lWhat size? \e[90m(Press #{up_down} arrow or 1-3 number to move and Enter to select)\e[0m\n", + "\e[?25lWhat size? \e[90m(Press #{up_down} arrow or 1-3 number to move and Space or Enter to select)\e[0m\n", " 1. large\n", "\e[32m#{symbols[:marker]} 2. medium\e[0m\n", " 3. small", @@ -208,7 +224,7 @@ def exit_message(prompt, choice) expect(value).to eq("Good choice!") expect(prompt.output.string).to eq([ - "\e[?25lWhat size? \e[90m(Press #{up_down} arrow or 1-3 number to move and Enter to select)\e[0m\n", + "\e[?25lWhat size? \e[90m(Press #{up_down} arrow or 1-3 number to move and Space or Enter to select)\e[0m\n", " 1) large\n", "\e[32m#{symbols[:marker]} 2) medium\e[0m\n", " 3) small", @@ -224,7 +240,7 @@ def exit_message(prompt, choice) prompt.input.rewind expect(prompt.select("What size?", choices, default: 2, enum: ".")).to eq("Medium") expect(prompt.output.string).to eq([ - "\e[?25lWhat size? \e[90m(Press #{up_down} arrow or 1-3 number to move and Enter to select)\e[0m\n", + "\e[?25lWhat size? \e[90m(Press #{up_down} arrow or 1-3 number to move and Space or Enter to select)\e[0m\n", " 1. Large\n", "\e[32m#{symbols[:marker]} 2. Medium\e[0m\n", " 3. Small", @@ -245,7 +261,7 @@ def exit_message(prompt, choice) expect(value).to eq("Large") expect(prompt.output.string).to eq([ - "\e[?25lWhat size? \e[31m(Press #{up_down} arrow to move and Enter to select)\e[0m\n", + "\e[?25lWhat size? \e[31m(Press #{up_down} arrow to move and Space or Enter to select)\e[0m\n", "\e[34m> Large\e[0m\n", " Medium\n", " Small", @@ -301,7 +317,7 @@ def exit_message(prompt, choice) expected_output = output_helper("What size?", choices.keys, :large, init: true, - hint: "Press #{up_down} arrow to move and Enter to select") + + hint: "Press #{up_down} arrow to move and Space or Enter to select") + "\e[?25h" expect(prompt.output.string).to eq(expected_output) @@ -320,7 +336,7 @@ def exit_message(prompt, choice) expected_output = output_helper("What size?", %w[Large Medium Small], "Large", init: true, - hint: "Press #{up_down} arrow to move and Enter to select") + + hint: "Press #{up_down} arrow to move and Space or Enter to select") + "\e[?25h" expect(prompt.output.string).to eq(expected_output) @@ -333,7 +349,7 @@ def exit_message(prompt, choice) prompt.input.rewind expect(prompt.select("What size?", choices)).to eq("Large") expect(prompt.output.string).to eq([ - "\e[?25l[?] What size? \e[90m(Press #{up_down} arrow to move and Enter to select)\e[0m\n", + "\e[?25l[?] What size? \e[90m(Press #{up_down} arrow to move and Space or Enter to select)\e[0m\n", "\e[32m#{symbols[:marker]} Large\e[0m\n", " Medium\n", " Small", @@ -353,11 +369,11 @@ def exit_message(prompt, choice) expected_output = output_helper("What size?", choices.keys, :large, init: true, - hint: "Press #{up_down} arrow to move and Enter to select") + + hint: "Press #{up_down} arrow to move and Space or Enter to select") + output_helper("What size?", choices.keys, :medium, - hint: "Press #{up_down} arrow to move and Enter to select") + + hint: "Press #{up_down} arrow to move and Space or Enter to select") + output_helper("What size?", choices.keys, :small, - hint: "Press #{up_down} arrow to move and Enter to select") + + hint: "Press #{up_down} arrow to move and Space or Enter to select") + exit_message("What size?", "small") expect(prompt.output.string).to eq(expected_output) @@ -393,7 +409,7 @@ def exit_message(prompt, choice) expect(answer).to eq("D") expected_output = [ - "\e[?25lWhat letter? \e[90m(Press #{up_down}/#{left_right} arrow to move and Enter to select)\e[0m\n", + "\e[?25lWhat letter? \e[90m(Press #{up_down}/#{left_right} arrow to move and Space or Enter to select)\e[0m\n", "\e[32m#{symbols[:marker]} D\e[0m\n", " E\n", " F", @@ -415,7 +431,7 @@ def exit_message(prompt, choice) expect(answer).to eq(4) expected_output = [ - "\e[?25lWhat letter? \e[90m(Press #{up_down}/#{left_right} arrow to move and Enter to select)\e[0m\n", + "\e[?25lWhat letter? \e[90m(Press #{up_down}/#{left_right} arrow to move and Space or Enter to select)\e[0m\n", "\e[32m#{symbols[:marker]} D\e[0m\n", " E\n", " F", @@ -440,7 +456,7 @@ def exit_message(prompt, choice) expect(answer).to eq("D") expected_output = [ - "\e[?25lWhat letter? \e[90m(Press #{up_down}/#{left_right} arrow to move and Enter to select)\e[0m\n", + "\e[?25lWhat letter? \e[90m(Press #{up_down}/#{left_right} arrow to move and Space or Enter to select)\e[0m\n", "\e[32m#{symbols[:marker]} D\e[0m\n", " E\n", " F", @@ -463,7 +479,7 @@ def exit_message(prompt, choice) expected_output = [ output_helper("What number?", choices[0..3], "1", init: true, - hint: "Press #{up_down}/#{left_right} arrow to move and Enter to select"), + hint: "Press #{up_down}/#{left_right} arrow to move and Space or Enter to select"), output_helper("What number?", choices[4..7], "5"), output_helper("What number?", choices[8..11], "9"), output_helper("What number?", choices[8..11], "9"), @@ -485,7 +501,7 @@ def exit_message(prompt, choice) expected_output = [ output_helper("What number?", choices[3..6], "4", init: true, - hint: "Press #{up_down}/#{left_right} arrow to move and Enter to select"), + hint: "Press #{up_down}/#{left_right} arrow to move and Space or Enter to select"), output_helper("What number?", choices[4..7], "8"), output_helper("What number?", choices[8..9], "10"), output_helper("What number?", choices[8..9], "10"), @@ -510,7 +526,7 @@ def exit_message(prompt, choice) expected_output = [ output_helper("What number?", choices[0..3], "2", init: true, - hint: "Press #{up_down}/#{left_right} arrow to move and Enter to select"), + hint: "Press #{up_down}/#{left_right} arrow to move and Space or Enter to select"), output_helper("What number?", choices[4..7], "6"), output_helper("What number?", choices[8..9], "10"), output_helper("What number?", choices[4..7], "6"), @@ -537,7 +553,7 @@ def exit_message(prompt, choice) expected_output = [ output_helper("What number?", choices[0..3], "2", init: true, - hint: "Press #{up_down}/#{left_right} arrow to move and Enter to select"), + hint: "Press #{up_down}/#{left_right} arrow to move and Space or Enter to select"), output_helper("What number?", choices[0..3], "3"), output_helper("What number?", choices[4..7], "7"), output_helper("What number?", choices[4..7], "6"), @@ -576,7 +592,7 @@ def exit_message(prompt, choice) expected_output = [ output_helper("What number?", choices[0..3], "1", init: true, - hint: "Press #{up_down}/#{left_right} arrow to move and Enter to select"), + hint: "Press #{up_down}/#{left_right} arrow to move and Space or Enter to select"), output_helper("What number?", choices[0..3], "3"), output_helper("What number?", choices[2..5], "5"), output_helper("What number?", choices[5..8], "8"), @@ -614,7 +630,7 @@ def exit_message(prompt, choice) expected_output = [ output_helper("What number?", choices[0..3], "2", init: true, - hint: "Press #{up_down}/#{left_right} arrow to move and Enter to select"), + hint: "Press #{up_down}/#{left_right} arrow to move and Space or Enter to select"), output_helper("What number?", choices[4..7], "7"), output_helper("What number?", choices[8..9], "9"), output_helper("What number?", choices[8..9], "9"), @@ -640,7 +656,7 @@ def exit_message(prompt, choice) expect(value).to eq("C") expected_output = [ output_helper("What letter?", choices, "A", init: true, - hint: "Press #{up_down} arrow to move and Enter to select"), + hint: "Press #{up_down} arrow to move and Space or Enter to select"), output_helper("What letter?", choices, "B"), output_helper("What letter?", choices, "C"), output_helper("What letter?", choices, "C"), @@ -660,7 +676,7 @@ def exit_message(prompt, choice) expect(answer).to eq("A") expected_output = [ output_helper("What letter?", choices, "A", init: true, - hint: "Press #{up_down} arrow to move and Enter to select"), + hint: "Press #{up_down} arrow to move and Space or Enter to select"), output_helper("What letter?", choices, "B"), output_helper("What letter?", choices, "C"), output_helper("What letter?", choices, "A"), @@ -685,7 +701,7 @@ def exit_message(prompt, choice) expected_output = output_helper("What letter?", choices, "B", init: true, - hint: "Press #{up_down} arrow to move and Enter to select") + + hint: "Press #{up_down} arrow to move and Space or Enter to select") + output_helper("What letter?", choices, "D") + output_helper("What letter?", choices, "B") + output_helper("What letter?", choices, "D") + @@ -709,7 +725,7 @@ def exit_message(prompt, choice) expected_output = [ output_helper("What number?", choices[0..3], "2", init: true, - hint: "Press #{up_down}/#{left_right} arrow to move and Enter to select"), + hint: "Press #{up_down}/#{left_right} arrow to move and Space or Enter to select"), output_helper("What number?", choices[4..7], "6"), output_helper("What number?", choices[8..9], "10"), output_helper("What number?", choices[0..3], "2"), @@ -747,7 +763,7 @@ def exit_message(prompt, choice) expected_output = [ output_helper("What number?", choices[0..3], "2", init: true, - hint: "Press #{up_down}/#{left_right} arrow to move and Enter to select"), + hint: "Press #{up_down}/#{left_right} arrow to move and Space or Enter to select"), output_helper("What number?", choices[4..7], "7"), output_helper("What number?", choices[8..9], "9"), output_helper("What number?", choices[0..3], "2"), @@ -774,7 +790,7 @@ def exit_message(prompt, choice) expected_output = output_helper("What room?", choices[0..3], "a2", init: true, - hint: "Press #{up_down}/#{left_right} arrow to move, Enter to select and letters to filter") + + hint: "Press #{up_down}/#{left_right} arrow to move, Space or Enter to select and letters to filter") + output_helper("What room?", choices[10..13], "b1", hint: "Filter: \"b\"") + output_helper("What room?", choices[14..17], "b5", hint: "Filter: \"b\"") + output_helper("What room?", choices[18..20], "b9", hint: "Filter: \"b\"") + @@ -838,7 +854,7 @@ def exit_message(prompt, choice) expect(value).to eq("A") expect(prompt.output.string).to eq([ - "\e[?25lWhat letter? \e[90m(Press #{up_down} arrow to move and Enter to select)\e[0m\n", + "\e[?25lWhat letter? \e[90m(Press #{up_down} arrow to move and Space or Enter to select)\e[0m\n", "\e[32m#{symbols[:marker]} A\e[0m\n", " B\n", " C\n", @@ -878,7 +894,7 @@ def exit_message(prompt, choice) expected_prompt_output = output_helper("What size?", %w[Small Medium Large Huge], "Small", init: true, - hint: "Press #{up_down} arrow to move, Enter to select and letters to filter") + + hint: "Press #{up_down} arrow to move, Space or Enter to select and letters to filter") + output_helper("What size?", %w[Medium Huge], "Medium", hint: "Filter: \"U\"") + output_helper("What size?", %w[Huge], "Huge", hint: "Filter: \"Ug\"") + exit_message("What size?", "Huge") @@ -886,6 +902,163 @@ def exit_message(prompt, choice) expect(actual_prompt_output).to eql(expected_prompt_output) end + it "filters and chooses entry with enter, with default key config" do + prompt.input << "g" << "\n" + prompt.input.rewind + + answer = prompt.select("What size?", %i[Small Medium Large Huge], filter: true) + expect(answer).to eql(:Large) + + actual_prompt_output = prompt.output.string + expected_prompt_output = + output_helper("What size?", %w[Small Medium Large Huge], "Small", init: true, + hint: "Press #{up_down} arrow to move, Space or Enter to select and letters to filter") + + output_helper("What size?", %w[Large Huge], "Large", hint: "Filter: \"g\"") + + exit_message("What size?", "Large") + + expect(actual_prompt_output).to eql(expected_prompt_output) + end + + it "filters and chooses entry with enter even if only return is passed in confirm_keys" do + prompt.input << "g" << "\n" + prompt.input.rewind + + answer = prompt.select("What size?", %i[Small Medium Large Huge], filter: true, confirm_keys: [:return]) + expect(answer).to eql(:Large) + + actual_prompt_output = prompt.output.string + expected_prompt_output = + output_helper("What size?", %w[Small Medium Large Huge], "Small", init: true, + hint: "Press #{up_down} arrow to move, Enter to select and letters to filter") + + output_helper("What size?", %w[Large Huge], "Large", hint: "Filter: \"g\"") + + exit_message("What size?", "Large") + + expect(actual_prompt_output).to eql(expected_prompt_output) + end + + it "filters and chooses entry with enter preserving the return key label" do + prompt.input << "g" << "\n" + prompt.input.rewind + + answer = prompt.select("What size?", %i[Small Medium Large Huge], filter: true, confirm_keys: [{return: ">ENTER<"}]) + expect(answer).to eql(:Large) + + actual_prompt_output = prompt.output.string + expected_prompt_output = + output_helper("What size?", %w[Small Medium Large Huge], "Small", init: true, + hint: "Press #{up_down} arrow to move, >ENTER< to select and letters to filter") + + output_helper("What size?", %w[Large Huge], "Large", hint: "Filter: \"g\"") + + exit_message("What size?", "Large") + + expect(actual_prompt_output).to eql(expected_prompt_output) + end + + it "filters and chooses entry with return and enter, with default key config" do + prompt.input << "g" << "\r\n" + prompt.input.rewind + + answer = prompt.select("What size?", %i[Small Medium Large Huge], filter: true) + expect(answer).to eql(:Large) + + actual_prompt_output = prompt.output.string + expected_prompt_output = + output_helper("What size?", %w[Small Medium Large Huge], "Small", init: true, + hint: "Press #{up_down} arrow to move, Space or Enter to select and letters to filter") + + output_helper("What size?", %w[Large Huge], "Large", hint: "Filter: \"g\"") + + exit_message("What size?", "Large") + + expect(actual_prompt_output).to eql(expected_prompt_output) + end + + it "filters and chooses entry with space, with default key config" do + prompt.input << "g" << " " + prompt.input.rewind + + answer = prompt.select("What size?", %i[Small Medium Large Huge], filter: true) + expect(answer).to eql(:Large) + + actual_prompt_output = prompt.output.string + expected_prompt_output = + output_helper("What size?", %w[Small Medium Large Huge], "Small", init: true, + hint: "Press #{up_down} arrow to move, Space or Enter to select and letters to filter") + + output_helper("What size?", %w[Large Huge], "Large", hint: "Filter: \"g\"") + + exit_message("What size?", "Large") + + expect(actual_prompt_output).to eql(expected_prompt_output) + end + + it "filters and chooses entry with Ctrl+S when configured as confirm key" do + prompt.input << "g" << "\C-s" + prompt.input.rewind + + answer = prompt.select("What size?", %i[Small Medium Large Huge], filter: true, confirm_keys: [:ctrl_s]) + expect(answer).to eql(:Large) + + actual_prompt_output = prompt.output.string + expected_prompt_output = + output_helper("What size?", %w[Small Medium Large Huge], "Small", init: true, + hint: "Press #{up_down} arrow to move, Ctrl+S to select and letters to filter") + + output_helper("What size?", %w[Large Huge], "Large", hint: "Filter: \"g\"") + + exit_message("What size?", "Large") + + expect(actual_prompt_output).to eql(expected_prompt_output) + end + + it "filters and chooses entry with Escape and custom label when configured as confirm key" do + prompt.input << "g" << "\e" + prompt.input.rewind + + answer = prompt.select("What size?", %i[Small Medium Large Huge], filter: true, confirm_keys: [escape: "Esc"]) + expect(answer).to eql(:Large) + + actual_prompt_output = prompt.output.string + expected_prompt_output = + output_helper("What size?", %w[Small Medium Large Huge], "Small", init: true, + hint: "Press #{up_down} arrow to move, Esc to select and letters to filter") + + output_helper("What size?", %w[Large Huge], "Large", hint: "Filter: \"g\"") + + exit_message("What size?", "Large") + + expect(actual_prompt_output).to eql(expected_prompt_output) + end + + it "filters and chooses entry with , and custom label when configured as confirm key" do + prompt.input << "g" << "," + prompt.input.rewind + + answer = prompt.select("What size?", %i[Small Medium Large Huge], filter: true, confirm_keys: [{escape: "Esc"}, {"," => "Comma (,)"}]) + expect(answer).to eql(:Large) + + actual_prompt_output = prompt.output.string + expected_prompt_output = + output_helper("What size?", %w[Small Medium Large Huge], "Small", init: true, + hint: "Press #{up_down} arrow to move, Esc or Comma (,) to select and letters to filter") + + output_helper("What size?", %w[Large Huge], "Large", hint: "Filter: \"g\"") + + exit_message("What size?", "Large") + + expect(actual_prompt_output).to eql(expected_prompt_output) + end + + it "does not choose entry with space, when configured to use only enter" do + prompt.input << "X" << " " << "X" << "\r" + prompt.input.rewind + + choices = ["X Large", "X X Large"] + + answer = prompt.select("What size?", choices, filter: true, confirm_keys: [:return]) + expect(answer).to eql("X X Large") + + actual_prompt_output = prompt.output.string + expected_prompt_output = + output_helper("What size?", choices, "X Large", init: true, + hint: "Press #{up_down} arrow to move, Enter to select and letters to filter") + + output_helper("What size?", choices, "X Large", hint: "Filter: \"X\"") + + output_helper("What size?", choices, "X Large", hint: "Filter: \"X \"") + + output_helper("What size?", ["X X Large"], "X X Large", hint: "Filter: \"X X\"") + + exit_message("What size?", "X X Large") + + expect(actual_prompt_output).to eql(expected_prompt_output) + end + it "filters and chooses the first of multiple matching entries" do prompt.input << "g" << "\r" prompt.input.rewind @@ -896,7 +1069,7 @@ def exit_message(prompt, choice) actual_prompt_output = prompt.output.string expected_prompt_output = output_helper("What size?", %w[Small Medium Large Huge], "Small", init: true, - hint: "Press #{up_down} arrow to move, Enter to select and letters to filter") + + hint: "Press #{up_down} arrow to move, Space or Enter to select and letters to filter") + output_helper("What size?", %w[Large Huge], "Large", hint: "Filter: \"g\"") + exit_message("What size?", "Large") @@ -913,7 +1086,7 @@ def exit_message(prompt, choice) actual_prompt_output = prompt.output.string expected_prompt_output = output_helper("What email?", %w[p*1@mail.com p*2@mail.com p*3@mail.com], "p*1@mail.com", init: true, - hint: "Press #{up_down} arrow to move, Enter to select and letters to filter") + + hint: "Press #{up_down} arrow to move, Space or Enter to select and letters to filter") + output_helper("What email?", %w[p*1@mail.com p*2@mail.com p*3@mail.com], "p*1@mail.com", hint: "Filter: \"p\"") + output_helper("What email?", %w[p*1@mail.com p*2@mail.com p*3@mail.com], "p*1@mail.com", hint: "Filter: \"p*\"") + output_helper("What email?", %w[p*2@mail.com], "p*2@mail.com", hint: "Filter: \"p*2\"") + @@ -935,7 +1108,7 @@ def exit_message(prompt, choice) actual_prompt_output = prompt.output.string expected_prompt_output = output_helper("What size?", %w[Tiny Medium Large Huge], "Tiny", init: true, - hint: "Press #{up_down} arrow to move, Enter to select and letters to filter") + + hint: "Press #{up_down} arrow to move, Space or Enter to select and letters to filter") + output_helper("What size?", %w[], "", hint: "Filter: \"z\"") + output_helper("What size?", %w[], "", hint: "Filter: \"z\"") + output_helper("What size?", %w[Large], "Large", hint: "Filter: \"a\"") + @@ -956,7 +1129,7 @@ def exit_message(prompt, choice) expected_prompt_output = output_helper("What size?", %w[Small Medium Large Huge], "Small", init: true, - hint: "Press #{up_down} arrow to move, Enter to select and letters to filter") + + hint: "Press #{up_down} arrow to move, Space or Enter to select and letters to filter") + output_helper("What size?", %w[Huge], "Huge", hint: "Filter: \"H\"") + output_helper("What size?", %w[Huge], "Huge", hint: "Filter: \"Hu\"") + output_helper("What size?", %w[Small], "Small", hint: "Filter: \"S\"") + @@ -979,7 +1152,7 @@ def exit_message(prompt, choice) expected_output = output_helper("What room?", choices[0..3], "a2", init: true, - hint: "Press #{up_down}/#{left_right} arrow to move, Enter to select and letters to filter") + + hint: "Press #{up_down}/#{left_right} arrow to move, Space or Enter to select and letters to filter") + output_helper("What room?", choices[10..13], "b1", hint: "Filter: \"b\"") + output_helper("What room?", choices[14..17], "b5", hint: "Filter: \"b\"") + output_helper("What room?", choices[18..20], "b9", hint: "Filter: \"b\"") + @@ -1005,7 +1178,7 @@ def exit_message(prompt, choice) expected_output = output_helper("What size?", choices, "Small", init: true, - hint: "Press #{up_down} arrow to move and Enter to select") + + hint: "Press #{up_down} arrow to move and Space or Enter to select") + output_helper("What size?", choices, "Medium") + output_helper("What size?", choices, "Huge") + "What size? \e[32mHuge\e[0m\n\e[?25h" @@ -1025,7 +1198,7 @@ def exit_message(prompt, choice) expected_output = output_helper("What letter?", choices, "A", init: true, - hint: "Press #{up_down} arrow to move, Enter to select and letters to filter") + + hint: "Press #{up_down} arrow to move, Space or Enter to select and letters to filter") + output_helper("What letter?", [], "", hint: "Filter: \"c\"") + output_helper("What letter?", [], "", hint: "Filter: \"c\"") + output_helper("What letter?", ["A"], "A", hint: "Filter: \"a\"") + @@ -1049,7 +1222,7 @@ def exit_message(prompt, choice) expected_output = output_helper("What size?", choices, "Small", init: true, enum: ") ", - hint: "Press #{up_down} arrow or 1-3 number to move and Enter to select") + + hint: "Press #{up_down} arrow or 1-3 number to move and Space or Enter to select") + output_helper("What size?", choices, "Small", enum: ") ") + "What size? \e[32mSmall\e[0m\n\e[?25h"