unicode_plot.cr

unicode_plot.cr

Test Lines of Code

Unicode terminal plots for Crystal, ported from Julia's UnicodePlots.jl.

🚧 UNDER CONSTRUCTION 🚧

  • Before the first release, history may be rewritten (force-push).
  • API and behavior are still being aligned with UnicodePlots.jl.

Current coverage

Implemented plot interfaces:

  • lineplot, lineplot!
  • scatterplot, scatterplot!
  • barplot
  • histogram
  • boxplot
  • densityplot, densityplot!
  • heatmap
  • spy
  • polarplot, polarplot!
  • stairs, stairs!
  • contourplot, contourplot! (experimental)

See examples/ for runnable scripts.

Installation

Add to your shard.yml:

dependencies:
  unicode_plot:
    github: kojix2/unicode_plot.cr

Then run shards install.

Quick start

require "unicode_plot"
include UnicodePlot

Line plot

x = (0..62).map { |i| i * Math::PI / 31.0 }
y = x.map { |v| Math.sin(v) }
puts lineplot(x, y, title: "sin(x)", xlabel: "x", ylabel: "sin(x)", color: :blue)

Scatter plot

x1 = (1..15).map { Random.rand * 3.0 + 1.0 }
y1 = (1..15).map { Random.rand * 3.0 + 1.0 }
x2 = (1..15).map { Random.rand * 3.0 + 6.0 }
y2 = (1..15).map { Random.rand * 3.0 + 6.0 }

p = scatterplot(x1, y1, name: "cluster A", color: :blue,
  title: "Two clusters", xlim: {0.0, 11.0}, ylim: {0.0, 11.0})
scatterplot!(p, x2, y2, name: "cluster B", color: :red)
puts p

Polar plot

theta = (0...40).map { |i| 4.0 * Math::PI * i.to_f64 / 39.0 }
r = theta.map { |angle| angle / (2.0 * Math::PI) }
puts polarplot(theta, r, title: "Polar line", color: :green)

Bar plot

cities  = ["Tokyo", "Delhi", "Shanghai", "São Paulo", "Mexico City"]
popmill = [13.96, 16.79, 24.18, 12.33, 9.21]
puts barplot(cities, popmill, title: "City populations", xlabel: "population [mil]")

You can also pass a Hash(String, Float64) directly (sorted by key):

puts barplot({"Ruby" => 95.0, "Python" => 98.0, "Crystal" => 72.0}, title: "Scores")

Histogram

data = (1..500).map { Random.rand * 10.0 }
puts histogram(data, title: "Uniform [0, 10)", nbins: 15)

Add series incrementally (! variants)

Most series-based plot types support a mutating ! variant:

x = (1..20).map(&.to_f)
p = lineplot(x, x.map { |v| Math.sqrt(v) }, name: "√x", color: :green, title: "Functions")
lineplot!(p, x, x.map { |v| Math.log(v) }, name: "ln(x)", color: :red)
puts p

Options

Option Description
title Plot title
xlabel / ylabel Axis labels
xlim / ylim Axis limits as {min, max} tuple
color Line/point color (:red, :blue, :green, :cyan, :magenta, :yellow, :normal)
name Series name (shown in legend)
xscale / yscale Scale function (:log2, :log10, or a Proc(Float64, Float64))
width / height Canvas size in characters
canvas Canvas type (:braille, :block, :ascii)

Practical notes

  • This project prioritizes Julia compatibility in visible output.
  • Some interfaces may still differ while ports are in progress.
  • spec/reference_spec.cr compares output against Julia reference assets.

Development

Run specs from repository root:

crystal spec

Reference fixtures are stored under spec/fixtures/.

License

MIT — see LICENSE.

Repository

unicode_plot.cr

Owner
Statistic
  • 0
  • 0
  • 0
  • 1
  • 0
  • about 9 hours ago
  • April 23, 2026
License

MIT License

Links
Synced at

Wed, 29 Apr 2026 10:56:27 GMT

Languages