YouPlot/lib/uplot/command.rb

393 lines
9.7 KiB
Ruby
Raw Normal View History

require 'optparse'
require 'csv'
2020-08-03 13:57:00 +08:00
require 'unicode_plot'
2020-07-29 16:01:39 +08:00
module Uplot
2020-08-15 21:12:42 +08:00
Data = Struct.new(:headers, :series)
2020-07-29 16:01:39 +08:00
class Command
2020-08-15 15:46:03 +08:00
Params = Struct.new(
2020-08-15 21:13:06 +08:00
# Sort me!
2020-08-15 15:46:03 +08:00
:title,
:width,
:height,
:border,
:margin,
:padding,
:color,
:xlabel,
:ylabel,
:labels,
:symbol,
:xscale,
:nbins,
:closed,
:canvas,
:xlim,
:ylim,
2020-08-15 21:13:06 +08:00
:grid,
:name
2020-08-15 15:46:03 +08:00
) do
def to_hc
to_h.compact
end
end
2020-08-15 16:10:41 +08:00
attr_accessor :params, :plot_type
2020-08-15 18:41:53 +08:00
attr_reader :raw_inputs
2020-08-15 16:10:41 +08:00
def initialize
2020-08-15 15:46:03 +08:00
@params = Params.new
2020-08-15 16:10:41 +08:00
@plot_type = nil
2020-08-03 13:57:00 +08:00
@headers = nil
2020-07-30 14:08:09 +08:00
@delimiter = "\t"
2020-07-31 16:35:21 +08:00
@transpose = false
2020-08-03 13:57:00 +08:00
@output = false
@count = false
@fmt = 'xyy'
2020-08-15 18:41:53 +08:00
@raw_inputs = []
@debug = false
2020-07-29 16:01:39 +08:00
end
2020-07-30 15:33:08 +08:00
def create_parser
2020-08-14 12:47:04 +08:00
OptionParser.new do |opt|
opt.program_name = 'uplot'
opt.version = Uplot::VERSION
2020-08-15 16:48:25 +08:00
opt.on('-o', '--output', TrueClass) do |v|
@output = v
end
.on('-d', '--delimiter VAL', String) do |v|
@delimiter = v
end
.on('-H', '--headers', TrueClass) do |v|
@headers = v
end
.on('-T', '--transpose', TrueClass) do |v|
@transpose = v
end
.on('-t', '--title VAL', String) do |v|
params.title = v
end
.on('-w', '--width VAL', Numeric) do |v|
params.width = v
end
.on('-h', '--height VAL', Numeric) do |v|
params.height = v
end
.on('-b', '--border VAL', Numeric) do |v|
params.border = v
end
.on('-m', '--margin VAL', Numeric) do |v|
params.margin = v
end
.on('-p', '--padding VAL', Numeric) do |v|
params.padding = v
end
.on('-c', '--color VAL', String) do |v|
params.color = v.to_sym
end
.on('-x', '--xlabel VAL', String) do |v|
params.xlabel = v
end
.on('-y', '--ylabel VAL', String) do |v|
params.ylabel = v
end
.on('-l', '--labels', TrueClass) do |v|
params.labels = v
end
.on('--fmt VAL', String) do |v|
@fmt = v
end
.on('--debug', TrueClass) do |v|
@debug = v
end
2020-08-14 12:47:04 +08:00
end
2020-07-30 11:44:31 +08:00
end
2020-07-29 16:01:39 +08:00
def parse_options(argv = ARGV)
2020-08-15 16:48:25 +08:00
main_parser = create_parser
parsers = Hash.new { |h, k| h[k] = create_parser }
parsers[:barplot] = \
parsers[:bar]
.on('--symbol VAL', String) do |v|
params.symbol = v
end
.on('--xscale VAL', String) do |v|
params.xscale = v
end
.on('--count', TrueClass) do |v|
@count = v
end
parsers[:count] = \
parsers[:c] # barplot -c
.on('--symbol VAL', String) do |v|
params.symbol = v
end
parsers[:histogram] = \
parsers[:hist]
.on('-n', '--nbins VAL', Numeric) do |v|
params.nbins = v
end
.on('--closed VAL', String) do |v|
params.closed = v
end
.on('--symbol VAL', String) do |v|
params.symbol = v
end
parsers[:lineplot] = \
parsers[:line]
.on('--canvas VAL', String) do |v|
params.canvas = v
end
.on('--xlim VAL', String) do |v|
params.xlim = get_lim(v)
end
.on('--ylim VAL', String) do |v|
params.ylim = get_lim(v)
end
parsers[:lineplots] = \
parsers[:lines]
.on('--canvas VAL', String) do |v|
params.canvas = v
end
.on('--xlim VAL', String) do |v|
params.xlim = get_lim(v)
end
.on('--ylim VAL', String) do |v|
params.ylim = get_lim(v)
end
parsers[:scatter] = \
parsers[:s]
.on('--canvas VAL', String) do |v|
params.canvas = v
end
.on('--xlim VAL', String) do |v|
params.xlim = get_lim(v)
end
.on('--ylim VAL', String) do |v|
params.ylim = get_lim(v)
end
parsers[:density] = \
parsers[:d]
.on('--grid', TrueClass) do |v|
params.grid = v
end
.on('--xlim VAL', String) do |v|
params.xlim = get_lim(v)
end
.on('--ylim VAL', String) do |v|
params.ylim = get_lim(v)
end
parsers[:boxplot] = \
parsers[:box]
.on('--xlim VAL', String) do |v|
params.xlim = get_lim(v)
end
# Preventing the generation of new sub-commands
parsers.default = nil
# Usage and help messages
main_parser.banner = \
<<~MSG
Program: uplot (Tools for plotting on the terminal)
Version: #{Uplot::VERSION} (using unicode_plot #{UnicodePlot::VERSION})
Usage: uplot <command> [options]
Command: #{parsers.keys.join(' ')}
Options:
MSG
2020-08-14 10:45:47 +08:00
begin
main_parser.order!(argv)
rescue OptionParser::ParseError => e
warn "uplot: #{e.message}"
exit 1
end
2020-08-15 16:10:41 +08:00
@plot_type = argv.shift&.to_sym
2020-07-29 16:01:39 +08:00
2020-08-15 16:10:41 +08:00
unless parsers.has_key?(plot_type)
if plot_type.nil?
2020-08-14 10:45:47 +08:00
warn main_parser.help
else
2020-08-15 16:10:41 +08:00
warn "uplot: unrecognized command '#{plot_type}'"
2020-08-14 10:45:47 +08:00
end
2020-07-30 11:44:31 +08:00
exit 1
end
2020-08-15 16:10:41 +08:00
parser = parsers[plot_type]
2020-08-14 10:45:47 +08:00
begin
parser.parse!(argv) unless argv.empty?
rescue OptionParser::ParseError => e
warn "uplot: #{e.message}"
exit 1
end
2020-07-30 10:48:27 +08:00
end
2020-08-15 21:17:44 +08:00
def get_lim(str)
str.split(/-|:|\.\./)[0..1].map(&:to_f)
end
2020-07-29 16:01:39 +08:00
def run
parse_options
2020-07-30 09:37:20 +08:00
# Sometimes the input file does not end with a newline code.
while input = Kernel.gets(nil)
input.freeze
2020-08-15 18:41:53 +08:00
@raw_inputs << input
2020-08-15 21:17:44 +08:00
@data = Preprocess.input(input, @delimiter, @headers, @transpose)
2020-08-15 16:10:41 +08:00
case plot_type
when :bar, :barplot
2020-08-15 21:17:44 +08:00
barplot(@data)
2020-08-15 16:10:41 +08:00
when :count, :c
2020-08-03 10:18:27 +08:00
@count = true
2020-08-15 21:17:44 +08:00
barplot(@data)
2020-08-15 16:10:41 +08:00
when :hist, :histogram
2020-08-15 21:17:44 +08:00
histogram(@data)
2020-08-15 16:10:41 +08:00
when :line, :lineplot
2020-08-15 21:17:44 +08:00
line(@data)
2020-08-15 16:10:41 +08:00
when :lines, :lineplots
2020-08-15 21:17:44 +08:00
lines(@data)
2020-08-15 16:10:41 +08:00
when :scatter, :scatterplot
2020-08-15 21:17:44 +08:00
scatter(@data)
2020-08-15 16:10:41 +08:00
when :density
2020-08-15 21:17:44 +08:00
density(@data)
2020-08-15 16:10:41 +08:00
when :box, :boxplot
2020-08-15 21:17:44 +08:00
boxplot(@data)
2020-08-15 16:10:41 +08:00
else
raise "unrecognized plot_type: #{plot_type}"
2020-07-30 09:37:20 +08:00
end.render($stderr)
2020-07-29 16:01:39 +08:00
2020-07-30 11:47:02 +08:00
print input if @output
2020-07-30 09:37:20 +08:00
end
2020-07-29 16:01:39 +08:00
end
2020-08-15 21:17:44 +08:00
def barplot(data)
headers = data.headers
series = data.series
if @count
series = Preprocess.count(series[0])
params.title = headers[0] if headers
2020-08-15 15:02:31 +08:00
end
2020-08-15 15:46:03 +08:00
params.title ||= headers[1] if headers
2020-08-15 21:17:44 +08:00
labels = series[0]
values = series[1].map(&:to_f)
UnicodePlot.barplot(labels, values, **params.to_hc)
2020-07-30 14:54:08 +08:00
end
2020-08-15 21:17:44 +08:00
def histogram(data)
headers = data.headers
series = data.series
params.title ||= data.headers[0] if headers
values = series[0].map(&:to_f)
UnicodePlot.histogram(values, **params.to_hc)
2020-07-31 12:41:12 +08:00
end
2020-08-15 21:17:44 +08:00
def line(data)
headers = data.headers
series = data.series
if series.size == 1
# If there is only one series, it is assumed to be sequential data.
2020-08-15 15:46:03 +08:00
params.ylabel ||= headers[0] if headers
2020-08-15 21:17:44 +08:00
y = series[0].map(&:to_f)
2020-08-15 15:46:03 +08:00
UnicodePlot.lineplot(y, **params.to_hc)
else
2020-08-15 21:17:44 +08:00
# If there are 2 or more series,
# assume that the first 2 series are the x and y series respectively.
if headers
params.xlabel ||= headers[0]
params.ylabel ||= headers[1]
end
x = series[0].map(&:to_f)
y = series[1].map(&:to_f)
2020-08-15 15:46:03 +08:00
UnicodePlot.lineplot(x, y, **params.to_hc)
2020-07-29 17:24:08 +08:00
end
end
2020-08-04 19:13:34 +08:00
def get_method2(method1)
(method1.to_s + '!').to_sym
2020-08-04 14:18:40 +08:00
end
2020-08-15 21:17:44 +08:00
def xyy_plot(data, method1)
headers = data.headers
series = data.series
2020-08-04 14:18:40 +08:00
method2 = get_method2(method1)
2020-08-15 21:17:44 +08:00
series.map! { |s| s.map(&:to_f) }
if headers
params.name ||= headers[1]
params.xlabel ||= headers[0]
end
params.ylim ||= series[1..-1].flatten.minmax # why need?
plot = UnicodePlot.public_send(method1, series[0], series[1], **params.to_hc)
2.upto(series.size - 1) do |i|
UnicodePlot.public_send(method2, plot, series[0], series[i], name: headers&.[](i))
2020-07-29 17:24:08 +08:00
end
plot
end
2020-07-30 14:54:08 +08:00
2020-08-15 21:17:44 +08:00
def xyxy_plot(data, method1)
headers = data.headers
series = data.series
2020-08-04 14:18:40 +08:00
method2 = get_method2(method1)
2020-08-15 21:17:44 +08:00
series.map! { |s| s.map(&:to_f) }
series = series.each_slice(2).to_a
2020-08-15 15:46:03 +08:00
params.name ||= headers[0] if headers
2020-08-15 21:17:44 +08:00
params.xlim = series.map(&:first).flatten.minmax # why need?
params.ylim = series.map(&:last).flatten.minmax # why need?
x1, y1 = series.shift
2020-08-15 15:46:03 +08:00
plot = UnicodePlot.public_send(method1, x1, y1, **params.to_hc)
2020-08-15 21:17:44 +08:00
series.each_with_index do |(xi, yi), i|
UnicodePlot.public_send(method2, plot, xi, yi, name: headers&.[]((i + 1) * 2))
2020-08-03 11:39:27 +08:00
end
plot
end
2020-08-15 21:17:44 +08:00
def lines(data)
2020-08-03 11:39:27 +08:00
case @fmt
when 'xyy'
2020-08-15 21:17:44 +08:00
xyy_plot(data, :lineplot)
2020-08-03 11:39:27 +08:00
when 'xyxy'
2020-08-15 21:17:44 +08:00
xyxy_plot(data, :lineplot)
2020-08-03 10:33:14 +08:00
end
end
2020-08-15 21:17:44 +08:00
def scatter(data)
2020-08-03 11:39:27 +08:00
case @fmt
when 'xyy'
2020-08-15 21:17:44 +08:00
xyy_plot(data, :scatterplot)
2020-08-03 11:39:27 +08:00
when 'xyxy'
2020-08-15 21:17:44 +08:00
xyxy_plot(data, :scatterplot)
2020-07-30 14:54:08 +08:00
end
end
2020-07-30 15:33:08 +08:00
2020-08-15 21:17:44 +08:00
def density(data)
2020-08-03 11:39:27 +08:00
case @fmt
when 'xyy'
2020-08-15 21:17:44 +08:00
xyy_plot(data, :densityplot)
2020-08-03 11:39:27 +08:00
when 'xyxy'
2020-08-15 21:17:44 +08:00
xyxy_plot(data, :densityplot)
2020-08-02 23:36:13 +08:00
end
end
2020-08-15 21:17:44 +08:00
def boxplot(data)
headers = data.headers
series = data.series
headers ||= (1..series.size).map(&:to_s)
series.map! { |s| s.map(&:to_f) }
UnicodePlot.boxplot(headers, series, params.to_hc)
2020-07-30 15:33:08 +08:00
end
2020-07-29 16:01:39 +08:00
end
end