170 lines
		
	
	
		
			3.8 KiB
		
	
	
	
		
			Ruby
		
	
	
	
	
	
			
		
		
	
	
			170 lines
		
	
	
		
			3.8 KiB
		
	
	
	
		
			Ruby
		
	
	
	
	
	
| # typed: strict
 | |
| # frozen_string_literal: true
 | |
| 
 | |
| # Various helper functions for interacting with TTYs.
 | |
| module Tty
 | |
|   @stream = T.let($stdout, T.nilable(T.any(IO, StringIO)))
 | |
| 
 | |
|   COLOR_CODES = T.let(
 | |
|     {
 | |
|       red:     31,
 | |
|       green:   32,
 | |
|       yellow:  33,
 | |
|       blue:    34,
 | |
|       magenta: 35,
 | |
|       cyan:    36,
 | |
|       default: 39,
 | |
|     }.freeze,
 | |
|     T::Hash[Symbol, Integer],
 | |
|   )
 | |
| 
 | |
|   STYLE_CODES = T.let(
 | |
|     {
 | |
|       reset:         0,
 | |
|       bold:          1,
 | |
|       italic:        3,
 | |
|       underline:     4,
 | |
|       strikethrough: 9,
 | |
|       no_underline:  24,
 | |
|     }.freeze,
 | |
|     T::Hash[Symbol, Integer],
 | |
|   )
 | |
| 
 | |
|   SPECIAL_CODES = T.let(
 | |
|     {
 | |
|       up:         "1A",
 | |
|       down:       "1B",
 | |
|       right:      "1C",
 | |
|       left:       "1D",
 | |
|       erase_line: "K",
 | |
|       erase_char: "P",
 | |
|     }.freeze,
 | |
|     T::Hash[Symbol, String],
 | |
|   )
 | |
| 
 | |
|   CODES = T.let(COLOR_CODES.merge(STYLE_CODES).freeze, T::Hash[Symbol, Integer])
 | |
| 
 | |
|   class << self
 | |
|     sig { params(stream: T.any(IO, StringIO), _block: T.proc.params(arg0: T.any(IO, StringIO)).void).void }
 | |
|     def with(stream, &_block)
 | |
|       previous_stream = @stream
 | |
|       @stream = T.let(stream, T.nilable(T.any(IO, StringIO)))
 | |
| 
 | |
|       yield stream
 | |
|     ensure
 | |
|       @stream = T.let(previous_stream, T.nilable(T.any(IO, StringIO)))
 | |
|     end
 | |
| 
 | |
|     sig { params(string: String).returns(String) }
 | |
|     def strip_ansi(string)
 | |
|       string.gsub(/\033\[\d+(;\d+)*m/, "")
 | |
|     end
 | |
| 
 | |
|     sig { params(line_count: Integer).returns(String) }
 | |
|     def move_cursor_up(line_count)
 | |
|       "\033[#{line_count}A"
 | |
|     end
 | |
| 
 | |
|     sig { params(line_count: Integer).returns(String) }
 | |
|     def move_cursor_up_beginning(line_count)
 | |
|       "\033[#{line_count}F"
 | |
|     end
 | |
| 
 | |
|     sig { params(line_count: Integer).returns(String) }
 | |
|     def move_cursor_down(line_count)
 | |
|       "\033[#{line_count}B"
 | |
|     end
 | |
| 
 | |
|     sig { returns(String) }
 | |
|     def clear_to_end
 | |
|       "\033[K"
 | |
|     end
 | |
| 
 | |
|     sig { returns(String) }
 | |
|     def hide_cursor
 | |
|       "\033[?25l"
 | |
|     end
 | |
| 
 | |
|     sig { returns(String) }
 | |
|     def show_cursor
 | |
|       "\033[?25h"
 | |
|     end
 | |
| 
 | |
|     sig { returns(T.nilable([Integer, Integer])) }
 | |
|     def size
 | |
|       return @size if defined?(@size)
 | |
| 
 | |
|       height, width = `/bin/stty size 2>/dev/null`.presence&.split&.map(&:to_i)
 | |
|       return if height.nil? || width.nil?
 | |
| 
 | |
|       @size = T.let([height, width], T.nilable([Integer, Integer]))
 | |
|     end
 | |
| 
 | |
|     sig { returns(Integer) }
 | |
|     def height
 | |
|       @height ||= T.let(size&.first || `/usr/bin/tput lines 2>/dev/null`.presence&.to_i || 40, T.nilable(Integer))
 | |
|     end
 | |
| 
 | |
|     sig { returns(Integer) }
 | |
|     def width
 | |
|       @width ||= T.let(size&.second || `/usr/bin/tput cols 2>/dev/null`.presence&.to_i || 80, T.nilable(Integer))
 | |
|     end
 | |
| 
 | |
|     sig { params(string: String).returns(String) }
 | |
|     def truncate(string)
 | |
|       (w = width).zero? ? string.to_s : (string.to_s[0, w - 4] || "")
 | |
|     end
 | |
| 
 | |
|     sig { returns(String) }
 | |
|     def current_escape_sequence
 | |
|       return "" if @escape_sequence.nil?
 | |
| 
 | |
|       "\033[#{@escape_sequence.join(";")}m"
 | |
|     end
 | |
| 
 | |
|     sig { void }
 | |
|     def reset_escape_sequence!
 | |
|       @escape_sequence = T.let(nil, T.nilable(T::Array[Integer]))
 | |
|     end
 | |
| 
 | |
|     CODES.each do |name, code|
 | |
|       define_method(name) do
 | |
|         @escape_sequence ||= T.let([], T.nilable(T::Array[Integer]))
 | |
|         @escape_sequence << code
 | |
|         self
 | |
|       end
 | |
|     end
 | |
| 
 | |
|     SPECIAL_CODES.each do |name, code|
 | |
|       define_method(name) do
 | |
|         @stream = T.let($stdout, T.nilable(T.any(IO, StringIO)))
 | |
|         if @stream&.tty?
 | |
|           "\033[#{code}"
 | |
|         else
 | |
|           ""
 | |
|         end
 | |
|       end
 | |
|     end
 | |
| 
 | |
|     sig { returns(String) }
 | |
|     def to_s
 | |
|       return "" unless color?
 | |
| 
 | |
|       current_escape_sequence
 | |
|     ensure
 | |
|       reset_escape_sequence!
 | |
|     end
 | |
| 
 | |
|     sig { returns(T::Boolean) }
 | |
|     def color?
 | |
|       require "env_config"
 | |
| 
 | |
|       return false if Homebrew::EnvConfig.no_color?
 | |
|       return true if Homebrew::EnvConfig.color?
 | |
|       return false if @stream.blank?
 | |
| 
 | |
|       @stream.tty?
 | |
|     end
 | |
|   end
 | |
| end
 | 
