unicode_plot.rb icon indicating copy to clipboard operation
unicode_plot.rb copied to clipboard

Lineplot has a limited number of colors to choose from.

Open kojix2 opened this issue 4 years ago • 8 comments

Barplot

UnicodePlot.lineplot([2,3,4],[5,6,7], title: "Bar", color: 32).render

OK.

Lineplot

UnicodePlot.lineplot([1,2,3], [4,5,8], color: 32).render

Error

        9: from /Library/Ruby/Gems/2.6.0/gems/unicode_plot-0.0.4/lib/unicode_plot/lineplot.rb:40:in `lineplot'
        8: from /Library/Ruby/Gems/2.6.0/gems/unicode_plot-0.0.4/lib/unicode_plot/lineplot.rb:85:in `lineplot!'
        7: from /Library/Ruby/Gems/2.6.0/gems/unicode_plot-0.0.4/lib/unicode_plot/grid_plot.rb:93:in `lines!'
        6: from /Library/Ruby/Gems/2.6.0/gems/unicode_plot-0.0.4/lib/unicode_plot/canvas.rb:159:in `lines!'
        5: from /Library/Ruby/Gems/2.6.0/gems/unicode_plot-0.0.4/lib/unicode_plot/canvas.rb:159:in `each'
        4: from /Library/Ruby/Gems/2.6.0/gems/unicode_plot-0.0.4/lib/unicode_plot/canvas.rb:160:in `block in lines!'
        3: from /Library/Ruby/Gems/2.6.0/gems/unicode_plot-0.0.4/lib/unicode_plot/canvas.rb:143:in `line!'
        2: from /Library/Ruby/Gems/2.6.0/gems/unicode_plot-0.0.4/lib/unicode_plot/braille_canvas.rb:49:in `pixel!'
        1: from /Library/Ruby/Gems/2.6.0/gems/unicode_plot-0.0.4/lib/unicode_plot/braille_canvas.rb:49:in `|'
TypeError (nil can't be coerced into Integer)

kojix2 avatar Dec 28 '20 07:12 kojix2

Awosome! @nanobowers

kojix2 avatar Jan 04 '21 00:01 kojix2

Hi @mrkn and @kojix2 , I think that getting proper support for 256 colors (and 24-bit) is necessary to fix this issue and to support heatmap plots (#29 ). As discussed in #36 , supporting numeric 256 colors without color mixing is insufficient.

per #29 :

We need to use https://github.com/red-data-tools/red-colors and https://github.com/red-data-tools/red-palette to implement colormap used in heatmap because this library is a backend of https://github.com/red-data-tools/charty.

I have looked some at red-colors and red-palette. I understand the reason to use them, but as they exist I believe they may be insufficient for handling ANSI colors without adding features from something like: https://github.com/janlelis/paint/blob/main/lib/paint.rb

Specifically, we need a way to detect or specify either 8/16/256/24-bit color modes, and support the conversion of color-spaces (e.g. RGB: "#C0FEFE") to/from the appropriate ANSI codes in any of the above color-modes. Some of this is part of red-colors, but some is not (Or I did not see it). Apparently 256 / 24-bit detection is not always easy due to differences in terminals and this can be hard to do well (e.g. differences in Mac vs. Linux, etc)

In any event, i believe there needs to be a decision on how to either

  • add some of the ANSI handling color-space code into UnicodePlot
  • add some of the ANSI handling color-space code into red-colors
  • add Paint (or other similar gem) as a dependency from red-colors or UnicodePlot

nanobowers avatar Jan 16 '21 17:01 nanobowers

I've create a pull-request to add Colors::Xterm256 in red-colors gem. I'll add color blending support in red-colors gem later.

We need to define how to blend the ANSI 16 colors and xterm's 256 colors in unicode_plot gem. In other words, we need to discuss how to support user-defined palette of ANSI 16 colors.

I think we can start without supporting user-defined ANSI 16 colors palette, but there are two options in this case:

  1. Use the orthodox palette for ANSI 16 colors.
  2. Forbid to blend ANSI 16 colors and xterm's 256 colors.

mrkn avatar Jan 17 '21 03:01 mrkn

@mrkn I am now running into this issue in trying to enable heatmap support. For heatmap, storing the colors as Colors::RGB objects is preferable compared to a single integer. This way the renderer can choose between output of xterm256 color or 24-bit output.

As you mention, it is unclear what to do here with regard to existing 16-color ansi support, especially with blending - one option I thought of is

  • if any color >16 is used, it is internally converted to an RGB object
  • if existing color and new color are both ansi-16, then preserve current behavior
  • if existing color and new color are both RGB objects, then blend those
  • if mixing RGB and ansi-16 then blend using Orthodox palette

The other thought I have is that with the lineplot case, the blending would happen successively and depending on the blending algorithm you may get different results. E.g. if the mixing used averaging applied in series, for one ordering of Red color data, would get average(average(130,70), 170) = 135, whereas another ordering average(average(130,170),70) = 110 Perhaps the blending would use max() which is not order dependent?

nanobowers avatar Feb 20 '21 20:02 nanobowers

So this issue is still going on...

kojix2 avatar May 30 '21 14:05 kojix2

@kojix2 I believe this is waiting on this pr: https://github.com/red-data-tools/red-colors/pull/13 Sadly, I forgot to mention this earlier.

nanobowers avatar May 30 '21 19:05 nanobowers

@nanobowers Thank you for reminding your pull-request.

@kojix2 As I mention in the pull-request, I need more time to find the appropriate distance function for deciding closest color.

mrkn avatar May 30 '21 23:05 mrkn

Moreover, we need to decide how to blend two or more colors. Maybe we need a way to specify the terminal background color.

mrkn avatar May 30 '21 23:05 mrkn