Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/piotrmurach/tty-reader
A set of methods for processing keyboard input in character, line and multiline modes.
https://github.com/piotrmurach/tty-reader
cli console line-editing ruby ruby-gem terminal tty word-completion
Last synced: 2 months ago
JSON representation
A set of methods for processing keyboard input in character, line and multiline modes.
- Host: GitHub
- URL: https://github.com/piotrmurach/tty-reader
- Owner: piotrmurach
- License: mit
- Created: 2017-08-30T18:33:59.000Z (over 7 years ago)
- Default Branch: master
- Last Pushed: 2023-11-09T11:18:46.000Z (about 1 year ago)
- Last Synced: 2024-11-15T20:15:27.342Z (2 months ago)
- Topics: cli, console, line-editing, ruby, ruby-gem, terminal, tty, word-completion
- Language: Ruby
- Homepage: https://ttytoolkit.org
- Size: 1010 KB
- Stars: 89
- Watchers: 3
- Forks: 17
- Open Issues: 12
-
Metadata Files:
- Readme: README.md
- Changelog: CHANGELOG.md
- Funding: .github/FUNDING.yml
- License: LICENSE.txt
- Code of conduct: CODE_OF_CONDUCT.md
Awesome Lists containing this project
- Open-Source-Ruby-and-Rails-Apps - tty-reader - A set of methods for processing keyboard input in character, line and multiline modes. π₯ β π (Happy Exploring π€)
README
# TTY::Reader [![Gitter](https://badges.gitter.im/Join%20Chat.svg)][gitter]
[![Gem Version](https://badge.fury.io/rb/tty-reader.svg)][gem]
[![Actions CI](https://github.com/piotrmurach/tty-reader/workflows/CI/badge.svg?branch=master)][gh_actions_ci]
[![Build status](https://ci.appveyor.com/api/projects/status/cj4owy2vlty2q1ko?svg=true)][appveyor]
[![Maintainability](https://api.codeclimate.com/v1/badges/2f68d5e8ecc271bda820/maintainability)][codeclimate]
[![Coverage Status](https://coveralls.io/repos/github/piotrmurach/tty-reader/badge.svg)][coverage]
[![Inline docs](http://inch-ci.org/github/piotrmurach/tty-reader.svg?branch=master)][inchpages][gitter]: https://gitter.im/piotrmurach/tty
[gem]: http://badge.fury.io/rb/tty-reader
[gh_actions_ci]: https://github.com/piotrmurach/tty-reader/actions?query=workflow%3ACI
[travis]: http://travis-ci.org/piotrmurach/tty-reader
[appveyor]: https://ci.appveyor.com/project/piotrmurach/tty-reader
[codeclimate]: https://codeclimate.com/github/piotrmurach/tty-reader/maintainability
[coverage]: https://coveralls.io/github/piotrmurach/tty-reader
[inchpages]: http://inch-ci.org/github/piotrmurach/tty-reader> A pure Ruby library that provides a set of methods for processing keyboard input in character, line and multiline modes. It maintains history of entered input with an ability to recall and re-edit those inputs. It lets you register to listen for keystroke events and trigger custom key events yourself.
**TTY::Reader** provides independent reader component for [TTY](https://github.com/piotrmurach/tty) toolkit.
![](assets/shell.gif)
## Compatibility
The `tty-reader` is not compatible with the GNU Readline and doesn't aim to be. It originated from [tty-prompt](https://github.com/piotrmurach/tty-prompt) project to provide flexibility, independence from underlying operating system and Ruby like API interface for creating different prompts.
`TTY::Reader` forges its own path to provide features necessary for building line editing in terminal applications!
## Features
* Pure Ruby
* Reading [single keypress](#21-read_keypress)
* [Line editing](#22-read_line)
* Reading [multiline input](#23-read_multiline)
* Ability to [register](#24-on) for keystroke events
* Track input [history](#32-track_history)
* No global state
* Works on Linux, OS X, FreeBSD and Windows
* Supports Ruby versions `>= 2.0.0` & JRuby## Installation
Add this line to your application's Gemfile:
```ruby
gem "tty-reader"
```And then execute:
$ bundle
Or install it yourself as:
$ gem install tty-reader
* [1. Usage](#1-usage)
* [2. API](#2-api)
* [2.1 read_keypress](#21-read_keypress)
* [2.2 read_line](#22-read_line)
* [2.3 read_multiline](#23-read_multiline)
* [2.4 on](#24-on)
* [2.5 subscribe](#25-subscribe)
* [2.6 unsubscribe](#26-unsubscribe)
* [2.7 trigger](#27-trigger)
* [2.8 supported events](#28-supported-events)
* [3. Configuration](#3-configuration)
* [3.1 :interrupt](#31-interrupt)
* [3.2 :track_history](#32-track_history)
* [3.3 :history_cycle](#33-history_cycle)
* [3.4 :history_duplicates](#34-history_duplicates)
* [3.5 :history_exclude](#35-history_exclude)
* [3.6 :history_size](#36-history_size)## Usage
In just a few lines you can recreate IRB prompt.
Initialize the reader:
```ruby
reader = TTY::Reader.new
```Then register to listen for key events, in this case listen for `Ctrl-X` or `Esc` keys to exit:
```ruby
reader.on(:keyctrl_x, :keyescape) do
puts "Exiting..."
exit
end
```Finally, keep asking user for line input with a `=>` as a prompt:
```ruby
loop do
reader.read_line("=> ")
end
```## API
### 2.1 read_keypress
To read a single key stroke from the user use `read_char` or `read_keypress`:
```ruby
reader.read_char
reader.read_keypress
reader.read_keypress(nonblock: true)
```### 2.2 read_line
By default `read_line` works in `raw mode` which means it behaves like a line editor that allows you to edit each character, respond to `control characters` such as `Control-A` to `Control-B` or navigate through history.
For example, to read a single line terminated by a new line character use `read_line` like so:
```ruby
reader.read_line
```If you wish for the keystrokes to be interpreted by the terminal instead, use so called `cooked` mode by providing the `:raw` option set to `false`:
```ruby
reader.read_line(raw: false)
```Any non-interpreted characters received are written back to terminal, however you can stop this by using `:echo` option set to `false`:
```ruby
reader.read_line(echo: false)
```You can also provide a line prefix displayed before input by passing it as a first argument:
```ruby
reader.read_line(">> ")
# >>
```To pre-populate the line content for editing use `:value` option:
```ruby
reader.read_line("> ", value: "edit me")
# > edit me
```### 2.3 read_multiline
By default `read_multiline` works in `raw mode` which means it behaves like a multiline editor that allows you to edit each character, respond to `control characters` such as `Control-A` to `Control-B` or navigate through history.
For example, to read more than one line terminated by `Ctrl+d` or `Ctrl+z` use `read_multiline`:
```ruby
reader.read_multiline
# => [ "line1", "line2", ... ]
```If you wish for the keystrokes to be interpreted by the terminal instead, use so called `cooked` mode by providing the `:raw` option set to `false`:
```ruby
reader.read_line(raw: false)
```You can also provide a line prefix displayed before input by passing a string as a first argument:
```ruby
reader.read_multiline(">> ")
```### 2.4 on
You can register to listen on a key pressed events. This can be done by calling `on` with a event name(s):
```ruby
reader.on(:keypress) { |event| .... }
```or listen for multiple events:
```ruby
reader.on(:keyctrl_x, :keyescape) { |event| ... }
```The `KeyEvent` object is yielded to a block whenever a particular key event fires. The event responds to:
* `key` - key pressed
* `value` - value of the key pressed
* `line` - the content of the currently edited line, empty otherwiseThe `value` returns the actual key pressed and the `line` the content for the currently edited line or is empty.
The `key` is an object that responds to following messages:
* `name` - the name of the event such as :up, :down, letter or digit
* `meta` - true if event is non-standard key associated
* `shift` - true if shift has been pressed with the key
* `ctrl` - true if ctrl has been pressed with the keyFor example, to add listen to vim like navigation keys, one would do the following:
```ruby
reader.on(:keypress) do |event|
if event.value == "j"
...
endif event.value == "k"
...
end
end
```You can subscribe to more than one event:
```ruby
reader.on(:keypress) { |event| ... }
.on(:keydown) { |event| ... }
```### 2.5 subscribe
You can subscribe any object to listen for the emitted [key events](#27-supported-events) using the `subscribe` message. The listener would need to implement a method for every event it wishes to receive.
For example, if a `MyListener` class wishes to only listen for `keypress` event:
```ruby
class MyListener
def keypress(event)
...
end
end
```Then subscribing is done:
```ruby
reader.subscribe(MyListener.new)
```Alternatively, `subscribe` allows you to listen to events only for the duration of block execution like so:
```ruby
reader.subscribe(MyListener) do
...
end
```### 2.6 unsubscribe
You can unsubscribe any object from listening to the key events using the `unsubscribe` message:
```ruby
reader.unsubscribe(my_listener)
```### 2.7 trigger
The signature for triggering key events is `trigger(event, args...)`. The first argument is a [key event name](#27-supported-events) followed by any number of actual values related to the event being triggered.
For example, to trigger `:keydown` event do:
```ruby
reader.trigger(:keydown)
```To add vim bindings for line editing you could discern between alphanumeric inputs like so:
```ruby
reader.on(:keypress) do |event|
if event.value == "j"
reader.trigger(:keydown)
end
if evevnt.value == "k"
reader.trigger(:keyup)
end
end
```### 2.8 supported events
The available key events for character input are:
* `:keypress`
* `:keyenter`
* `:keyreturn`
* `:keytab`
* `:keybackspace`
* `:keyspace`
* `:keyescape`
* `:keydelete`
* `:keyalpha`
* `:keynum`The navigation related key events are:
* `:keydown`
* `:keyup`
* `:keyleft`
* `:keyright`
* `:keyhome`
* `:keyend`
* `:keyclear`The specific `ctrl` key events:
* `:keyctrl_a`
* `:keyctrl_b`
* ...
* `:keyctrl_z`The key events for functional keys `f*` are:
* `:keyf1`
* `:keyf2`
* ...
* `:keyf24`## 3. Configuration
### 3.1. `:interrupt`
By default `InputInterrupt` error will be raised when the user hits the interrupt key(Control-C). However, you can customise this behaviour by passing the `:interrupt` option. The available options are:
* `:signal` - sends interrupt signal
* `:exit` - exists with status code
* `:noop` - skips handler
* custom procFor example, to send interrupt signal do:
```ruby
reader = TTY::Reader.new(interrupt: :signal)
```### 3.2. `:track_history`
The `read_line` and `read_multiline` provide history buffer that tracks all the lines entered during `TTY::Reader.new` interactions. The history buffer provides previous or next lines when user presses up/down arrows respectively. However, if you wish to disable this behaviour use `:track_history` option like so:
```ruby
reader = TTY::Reader.new(track_history: false)
```### 3.3. `:history_cycle`
This option determines whether the history buffer allows for infinite navigation. By default it is set to `false`. You can change this:
```ruby
reader = TTY::Reader.new(history_cycle: true)
```### 3.4. `:history_duplicates`
This option controls whether duplicate lines are stored in history. By default set to `false`. You can change this:
```ruby
reader = TTY::Reader.new(history_duplicates: true)
```### 3.5. `:history_exclude`
This option allows you to exclude lines from being stored in history. It accepts a `Proc` with a line as a first argument. By default it is set to exclude empty lines. To change this:
```ruby
reader = TTY::Reader.new(history_exclude: ->(line) { ... })
```### 3.6. `:history_size`
By default, the history buffer can store up to `512` lines. This can be changed with the `:history_size` configuration:
```ruby
reader = TTY::Reader.new(history_size: 2048)
```## Development
After checking out the repo, run `bin/setup` to install dependencies. Then, run `rake spec` to run the tests. You can also run `bin/console` for an interactive prompt that will allow you to experiment.
## Contributing
Bug reports and pull requests are welcome on GitHub at https://github.com/piotrmurach/tty-reader. This project is intended to be a safe, welcoming space for collaboration, and contributors are expected to adhere to the [Contributor Covenant](http://contributor-covenant.org) code of conduct.
1. Clone the project on GitHub
2. Create a feature branch
3. Submit a Pull RequestImportant notes:
- **All new features must include test coverage.** At a bare minimum, unit tests are required. It is preferred if you include acceptance tests as well.
- **The tests must be be idempotent.** Any test run should produce the same result when run over and over.
- **All new features must include source code & readme documentation** Any new method you add should include yarddoc style documentation with clearly specified parameter and return types.## License
The gem is available as open source under the terms of the [MIT License](http://opensource.org/licenses/MIT).
## Code of Conduct
Everyone interacting in the TTY::Reader projectβs codebases, issue trackers, chat rooms and mailing lists is expected to follow the [code of conduct](https://github.com/piotrmurach/tty-reader/blob/master/CODE_OF_CONDUCT.md).
## Copyright
Copyright (c) 2017 Piotr Murach. See LICENSE for further details.