-
Notifications
You must be signed in to change notification settings - Fork 19
Plot Widget
Drawing a graph or a chart or a plot? Shoes 3.3.2 (beta r2649 or better) has a plot widget. It is not a replacement for programs like gnuplot, LibreOffice, OpenOffice, or other professional quality programs. Shoes has a limited set of options and they don't alway work as you might expect. You can always export your data to csv and use a better program to display the data.
Now that you know what to do when disappointed. Let's have some fun!
Assume we have an array of numbers and we want to plot them.
Shoes.app width: 620, height: 500 do
@values1 = [24, 22, 10, 13, 20, 8, 22]
@grf = plot 600, 400, title: "My Graph", caption: "my caption could be long",
font: "Helvetica"
@grf.add values: @values1, name: "foobar", min: 6, max: 26
end
That is about as minimal as we can get. It looks like this:
You'll notice that there are two parts. Creating the plot widget and saving it as @grf and adding the array to that @grf. You can create the widget with out adding a data series to display so in truth, you could use @grf = plot 600, 400, {}
but that's boring and the fun is in the {options}. You can graph multiple data series (up to 6 for some chart types). The {options} for add are important and not so obvious. I call them "data series" because there is more information needed than just a Ruby array of values.
The min: and max: options could be calculated for you but aren't because it gives you some control on how the graph looks. So much control, that you shouldn't use defaults, so are there are none.
What if you wanted your own x-axis labels instead of the generated 1,2,3...?
Shoes.app width: 620, height: 500 do
@values1 = [24, 22, 10, 13, 20, 8, 22]
@x_axis1 = ['a','b','c','d','e','f', 'g']
@grf = plot 600, 400, title:"My Graph", caption: "my caption could be long",
font: "Helvetica"
@grf.add values: @values1, labels: @x_axis1,
name: "foobar", min: 6, max: 26
end
It's a major mistake if you don't have enough strings in the x observations (labels: array) as you do in the values: array. Since labels[] are strings they could be anything. Perhaps a Date or Time string? - up to you. You should have the same number of labels[] and values[]. Shoes will auto-size the data points (observations) to the range given. I could have used min: @values1.min, max: @values1.max
but specifying my own range is a very useful thing if your plot has two data series attached with very different scales.
Note:, don't be tempted to use Ruby's min and max functions to compute the chart options. They don't like nil values which some chart types can handle.
Inside Shoes, each plot.add adds to an array so the first data series is 0 and the second is 1. Give the data series a name: "string". Mandatory. It displays in the legend part of the box.
The first data series added to the plot also controls what is displayed in the x-axis (there is only one of those) and will control the number of ticks marks and labels drawn on the left side and right side y-axis. Ticks? I don't see any ticks? Add auto_grid: true
to the plot creation.
@grf = plot 600, 400, title: "My Graph", caption: "my caption could be long",
font: "Helvetica", auto_grid: true
Let's use many of the options for plot and add options with a program that creates two data series, one of which has a missing value (it's nil) and an label is nil too.
See gr1.rb and comment/uncomment the settings for which set of values an labels you'd like to see.
That example also shows some more options when creating a plot.
chart: "timeseries",
default: "skip", click: proc {|btn, l, t| puts "click on #{@grf.near_x(l)}" }
There are several chart types - the default is "line". "Column", Scatter" and "Pie". "timeseries" is like "line" on steroids. Not all options apply to all chart types and in this case only line and timeseries allow for a click: and default: only works on some chart types.
From the gr1.rb example you can see a new option when adding a dataseries to the plot,
desc: "foobar Yy"
That (desc:) is displayed in the legend and colored to match the data series. name: is used if desc: isn't given. name: is mandatory, desc: is optional.
There are two other options for the plot widget creation. Shoes doesn't know the pixel width of the 'typical' label string. By default it's somewhere around 6 characters but you do get situations where the auto_sizing needs help (depends on your data and strings, the size of the widget....). So { x_ticks: 4, y_ticks: 20} would only draw four vertical lines on the grid and 20 horizontal lines, for example. This also introduces it's own visual artifacts so you'll need to play with what works best for you and of course it only applies to plot/chart types that accept them.
Wait, there's more options you can use when adding a data series. If you don't like the default colors, you can specify your own using the predefined names in Shoes (Manual->Shoes->Color) when you add a data series to the plot with the color: "shoes_color_string" option.
You can also set a wider line width for a data series with { strokewidth: small-int }. Anything less that one will be set to one, otherwise it wouldn't be visible.
You might notice in the picture there is some sort of dot drawn at each data point in the picture above. That's option { points: boolean }. It only looks like a circle - it's really a wee small '+' or box. The points , even if specified will only show up if Shoes thinks it has enough space for the given number of data points (not that many - if width can handle 1/10 of the data - you can get points in a line or timeseries). A data series with lots of observations (values) will turn off the points display. It just makes it slightly easier to visually find the point and figure out the data value. In some charts (scatter) you can specify "dot", "box", "circle", "rect". Dot and box are filled in and circle and rect aren't.
They aren't the same except for a small number of observations. I've got timeseries data files with 4K points which doesn't fit in a 600 pixel width widget. Yet Shoes does draw all the points, you just can't see them. That is true for all charts - they draw everything given in the space they have.
Timeseries charts have some special features. You set the display indices for the left side and right side. That allows you to scroll left and right in the display, zoom it and zoom out if you write the code to handle that.
See grcsv.rb
Methods set_first and set_last change the display of data points. It does not change the size of the values and label arrays. They change the display of every data series on the plot/graph. You use them to zoom in, out and scroll left or right. It's a lot work for you but if you need it, you can do it.
redraw_to(index) allows you to append/change data in your values and labels, in a line chart or timeseries chart and then tell Shoes to draw everything to that index Perhaps you are collecting a data value in real time from a sensor or some web site. IT IS VERY IMPORTANT that you must append the data to the values array and to the labels array BEFORE calling redraw_to. If you're managing zooming or scrolling, you need to know that it will reset the begin and end display indexes to the begining ([0]) and end (essentially unzoom, unscroll). Manage your scroll/zoom wisely. You have the tools.
Again see that gr1.rb test.
You can save the plot (aka) chart as a png, pdf, ps or svg. Just make sure the filename extension is one of those four lower case, strings following the last .
- it does parse the extension. The vector formats like .pdf are a lot of fun because you can stretch them or shrink them without pixel jaggies. It's Impressive. There might be a performance or file size issue with a 6 data series of 5000 data points each drawn on the screen. They draw just fine, but I haven't tried saving that to a vector fix because since I can't fix whatever might be wrong with the file then I haven't tried.
Shoes doesn't do horizontal "bar" charts or stacked columns or stacked bars -- life is short.
You can fill the background with a Shoes color name. While the default colors can be used, you probably want one from the Shoes pallet.
In the option hash, you can specify chart: "column"
(default is chart: "line"
). Here's two plot widgets
source: gr3.rb:
Looking at the code, it draws the same data into two different plots widgets, one line chart and one column chart.
If you're paying attention, you'll notice there is no boundary box drawn around the column chart (aka plot). It's an option {boundary_box: boolean}
, default is `:true".
A note or two about column graphs: It would be your mistake to cram too many bars in a small space. strokewidth: is how you specify the width of column. Refer to that Shoes code link. If a column value matches the given minimum value, there is no column to draw so set your min and max by hand. Just a friendly warning if you think it's not displaying a value. Been there, done that.
gr2.rb shows how Shoes line chart can handle missing data points and missing observation labels. Top Tip: Fix your data and don't depend on Shoes drawing what you think it should do if your data is wrong.
The points: hash argument can now take a string argument as well as a :true, :false, or nil. The string argument is one of "dot", "circle", "box", or "rect". Dot and box are filled with series color and Circle and Box are hollow so the background shows inside. The size of of the dot/circle/box/rect is related to or controlled by the strokewidth: setting and they only become visible if Shoes thinks the chart is "not too busy" and only for line, timeseries and scatter graphs.
Scatter charts are a little cumbersome and come with some odd rules and behavior in Shoes.
The first rule is that it takes two @grf.add {series} - no more and no less than two. It ignores the labels: of both data series. The first series added is assumed to be the x (horizontal) values and the second series added is the y (vertical) values.
The second rule is that you really should specify the minv and maxv values for both series and set them wisely. Sadly, wisely doesn't have any rules-of-thumb because Shoes chart auto_sizing rules are mysterious.
You'll also notice that a scatter plot has a different legend - the x-axis label is drawn under the x axis and the y-axis (second series) is drawn vertically on the left. You might notice that compared to a line or bar chart there is a smaller graph area to allow that.
Pie charts also have their own rules. First rule - only one data series can be added. Exactly one. And the x-axis labels (strings) are used to build the legend, which is displayed near the right-top. Auto-grid :true means to draw a box around the legend (not the chart space).
One other thing you can do on a pie chart is specify **pie_percent:** true
if you want the labels to be a percentage instead of their value.