Gattino.jl

Elegant data visualization for julia
Author ChifiSource
Popularity
11 Stars
Updated Last
8 Months Ago
Started In
April 2021

gattino

Gattino is Toolips-based, extensible plotting for Julia. Gattino features ...

  • composable, fully-featured data-visualization
  • an extension ecosystem
  • toolips compatibility
  • powerful layout syntax

There is currently a lot underway when it comes to Chifi packages, so this package is currently a work in progress.

map

getting started

adding gattino

Before Gattino is merged to the Julia General Pkg registry, `Gattino will need to be added by URL.

using Pkg; Pkg.add(url = "https://github.com/ChifiSource/Gattino.jl")
using Gattino

If you would like to use the Unstable version of Gattino, which will have more features but be less stable, set the rev key-word argument to Unstable.

using Pkg; Pkg.add(url = "https://github.com/ChifiSource/Gattino.jl", rev = "Unstable")
using Gattino
resources

chifi is currently working an in-ecosystem Olive-based documentation (and notebook) webapp which will hold the documentation for this project as well as other modules from this organization. While this new interactive documentation is still in the works, the resources for information on Gattino will be limited to

Fortunately, we have a lot of plans for resources coming in the future and if this README is on the main branch it probably means that these plans are pretty well in motion; Gattino is meant to be coming at around the same time as these new resources.

visualizations

creating visualizations

While Gattino plots are completely composable and can be made by composing context plotting elements together, the module also comes with some high-level functions which may be used to produce standard visualizations we are likely familiar with. These examples currently include scatter, line, and hist.

scatter(x::Vector, y::Vector, divisions::Int64 = 4, title::String = "")

line(x::Vector, y::Vector, divisions::Int64 = 4, title::String = "")

hist(x::Vector, y::Vector, divisions::Int64 = 4, title::String = "")

These methods are used to create a minimalistic visualization which can be further mutated with other Gattino methods. In each of these examples,the return type will be an AbstractContext. Let's make our first histogram with Gattino.

myhist = hist(["emma", "emmy", "em"], [22, 25, 14], title = "votes for names")

The hist function is just a passthrough to hist!, which is a context plotting function that creates a histogram. Notably, hist will create a new Context for us and hist! expects us to provide an AbstractContext as an argument. The line and scatter equivalence to this is found in scatter_plot! and line_plot!. That being said, if we want to add a visualization to a Context that already exists, we would use these methods, rather than the high-level method. When using hist! we will want to add our histogram to an old plot, when using hist we will be making a new plot with a new window. Here, I will use the context and group! functions to compose a scatter with the scatter_plot! method.

myframe = context(500, 500) do con::Context
    group!(con, "scatter", 250, 250) do g::Group
        Gattino.scatter_plot!(g, [1, 2, 3, 4], [1, 2, 3, 4])
    end
end

Graphics in Gattino are scaled using the Context and Group types. A Context represents a window, whereas a Group represents an area in that window. To create a context, we use the context method.

  • context(f::Function, width::Int64 = 1280, height::Int64= 720, margin::Pair{Int64, Int64} = 0 => 0)
mycontext = context(500, 500) do con::Context

end

This Context can now be used with context plotting methods. There are two different types of group which we can use on our project, group!is the mutating group -- this will add anything drawn to the group to theContext`.

  • group is non-mutating group -- anything we draw will not be drawn onto the window.

The methods are

  • group(f::Function, c::AbstractContext, w::Int64, h::Int64, margin::Pair{Int64, Int64})
  • `group!(f::Function, c::AbstractContext, name::String, w::Int64, h::Int64, margin::Pair{Int64, Int64})

These dispatches are for the most part the same as the context method. The width, height, and margin will all default to those of the provided AbstractContext. Additionally, group! will take the name of the layer as the second positional argument. These two forms of group are used in tandem to organize the layers of our Context. group is used to define new AbstractContext dimensions without adding a layer, whereas group! will add a new layer in those dimensions. For example.

mycon = context(500, 500) do con::Context
    group(con, 500, 250) do gridbox::Group
        group!(gridbox, "grid") do g::Group
            Gattino.grid!(g, 4)
        end
    end
    group(con, 500, 250, 0 => 250) do otherbox::Group
        group!(otherbox, "grid2") do g::Group
            Gattino.grid!(g, 4, "stroke" => "pink")
            
        end
    end
    Gattino.text!(con, 230, 250, "hello!")
end

In this case, I used group to create an initial AbstractContext with a certain dimensionality so that we could draw a bunch of things onto it. Note the use of group in this case, as I do not want this group to be drawn as a layer it is only used to change the dimensions. Next, I used group! whenever I actually wanted to draw onto the grid. The advantage to using group! like this is that we get all of the elements on different layers. We can access these layers with the layers function.

layers(mycon)

3-element Vector{Pair{Int64, String}}:
 1 => "grid"
 2 => "grid2"
 3 => "la81WFbV"
layouts

With the last example, we got an idea of how we might stack plots on top of one another. We have two different options which may be used to create layouts. The first of these options was demonstrated prior, this is using the margin, height, and width arguments with Groups to draw scaled frames in different portions of our window. Our width, height, and margins are provided to either the context method or one of the group methods (group/group!) in that order. Our margins push our frame to the right or down as they increase.

myframe = context(500, 250) do con::Context
    group(con, 250, 250) do g::Group
        Gattino.scatter_plot!(g, [1, 2, 3, 4], [1, 2, 3, 4])
    end
end

In this case, we have a Context, or window, of width 500 and height 250. The group we created below this is of width 250 and of height 250 -- the full height and half of the width. Let's add a grid with a margin of 250 on the X with the same size. This will make it easier to discern the difference between these visualizations.

group(myframe, 250, 250, 250 => 0) do g::Group
    Gattino.grid!(g, 4, "stroke" => "pink")
end

This form of layouts is for the inside of a Context. We also have the option to form layouts by placing Contexts next to eachother. To do this, use the compose function. This takes a name, the name of the window we want to create, and then our Context(s).

plt = context(200, 200) do con::Context
    group!(con, "points") do g::Group
        Gattino.points!(g, [5, 10, 15], [5, 10, 15])
    end
end
plt2 = context(200, 200) do con::Context
    group!(con, "points") do g::Group
        Gattino.points!(g, [5, 10, 15], [5, 10, 15], "fill" => "blue")
    end
end
n = compose("new", plt2, plt)

With this technique, we are also able to add different types of contexts together, and this is very useful for making full dashboard layouts with very little effort. We have three main functions for this,

hcat
push!
vcat

Concatenating horizontally will add the Context horizontally, vertically will add it vertically and push! will of course also concatenate horizontally.

vcat(n, plt, plt2)

working with layers

An important aspect to Gattino is the layering aspect. In Gattino, visualizations are premade from the context plotting toolkit and then mutated by making changes to the layers. We are able to access the layers of an AbstractContext using the layers method.

layers(con::AbstractContext)
using Gattino
myvis = Gattino.scatter([10, 4, 3, 4, 6, 4], [10, 3, 5, 3, 4, 4])
layers(myvis)

5-element Vector{Pair{Int64, String}}: 1 => "axes" 2 => "grid" 3 => "points" 4 => "labels" 5 => "axislabels"

We can also index a Context with a String to retrieve a layer directly.

getindex(con::AbstractContext, str::String)
myvis["points"]

We can also mutate layers using the various methods Gattino provides to do so. These include

  • style!(con::AbstractContext, s::String, spairs::Pair{String, String} ...)
  • move_layer!(con::Context, layer::String, to::Int64)
  • delete_layer!(con::Context, layer::String)
  • merge!(c::AbstractContext, c2::AbstractContext)
  • open_layer!(f::Function, con::AbstractContext, layer::String)
styling layers

The first thing we are going to want to do with our new Gattino visualization is probably style it, for this we use the following style dispatch:

  • style!(con::AbstractContext, s::String, spairs::Pair{String, String} ...)

Let's get our histogram back from earlier.

myhist = Gattino.hist(["emma", "emmy", "em"], [22, 25, 14], title = "votes for names")

In order to use this dispatch, we will need to provide a layer name as the second argument. In order to check the layers currently in your Context, use layers(::AbstractContext). Let's try this on the histogram we created.

layers(myhist)
7-element Vector{Pair{Int64, String}}:
1 => "XS3ms8yi"
2 => "title"
3 => "axes"
4 => "grid"
5 => "bars"
6 => "labels"
7 => "axislabels"

We are able to style these all individually with the style! dispatch we created earlier. These style! calls are simply CSS pairs from Toolips. Let's change the fill of our bars and make some other adjustmnets to the labels.

style!(myhist, "bars", "fill" => "orange", "opacity" => 70percent)
style!(myhist, "labels", "stroke-width" => 0px, "fill" => "white", "font-weight" => "bold")

We may also style our window itself with

style!(myhist, "border" => "5px solid black")
editing layers

Editing layers will primarily consist of either setting the attributes of layers using open_layer!, or using one of the following to mutate the state of the layers:

  • set!(ecomp::Pair{Int64, <:Toolips.Servable}, prop::Symbol, value::Any)
  • set!(ecomp::Pair{Int64, <:Toolips.Servable}, prop::Symbol, vec::Vector{<:Number}; max::Int64 = 10)
  • move_layer!(con::AbstractContext, layer::String, to::Int64)
  • delete_layer!(con::Context, layer::String)
  • merge!(con::Context, othercon::Context)

move_layer! and delete_layer! are both straightforward.

layers(myhist)
7-element Vector{Pair{Int64, String}}: 1 => "4qGRocCq" 2 => "title" 3 => "axes" 4 => "grid" 5 => "bars" 6 => "labels" 7 => "axislabels"

move_layer!(myhist, "axislabels", 2)

7-element Vector{Pair{Int64, String}}: 1 => "4qGRocCq" 2 => "axislabels" 3 => "title" 4 => "axes" 5 => "grid" 6 => "bars" 7 => "labels"

delete_layer!(myhist, "axislabels")


6-element Vector{Pair{Int64, String}}: 1 => "4qGRocCq" 2 => "title" 3 => "axes" 4 => "grid" 5 => "bars" 6 => "labels"

merge! is used to merge to Contexts together. This will essentially just copy the shapes, and does no scaling.

Gattino.merge!(myhist, myvis)
setting attributes

The final piece of this puzzle is the ability to mutate the properties of each Component in a layer. This is done with set! functions, which will take a Pair with our layer element and its enumeration. For many of these functions, we are also able to add a Vector to change these properties based on the values of a continuous feature.

  • set!(ecomp::Pair{Int64, <:Toolips.Servable}, prop::Symbol, to::Any)
  • set!(ecomp::Pair{Int64, <:Toolips.Servable}, prop::Symbol, vec::Vector{<:Number}; max::Int64 = 10)
  • style!(ecomp::Pair{Int64, <:Toolips.AbstractComponent}, vec::Vector{<:Number}, stylep::Pair{String, Int64} ...)
  • set_gradient!(ecomp::Pair{Int64, <:Toolips.Servable}, vec::Vector{<:Number}, colors::Vector{String})
  • set_shape!(ecomp::Pair{Int64, <:Toolips.Servable}, shape::Symbol)

To get started, we will create a visualization and then open a layer.

mycon = context(500, 500) do con::Context
    Gattino.scatter_plot!(con, firstfeature, secondfeature)
end

Gattino.open_layer!(mycon, "points") do ec

end

Here, I will use style!, set_gradient!, and set! to show three different features:

Gattino.open_layer!(mycon, "points") do ec
     Gattino.set!(ec, :r, thirdfeature, max = 60)
     style!(ec, fourth_feature, "stroke-width" => 10)
     Gattino.set_gradient!(ec, fourth_feature)
end
annotations

Because Gattino plot features are composable, we are able to easily annotate Gattino plots

animation

Animation in Gattino centers around Toolips' Animation syntax. To create an animation, we will call the Animation constructor, like so:

myanim = Animation("myanim", iterations = 1, length = 1.1)

Now we set the steps of our animation, this can be done with percentages or :from and :to:

myanim[:from] = "opacity" => 0percent
myanim[:from] = "transform" => "scale(0)"
myanim[:to] = "opacity" => 100percent
myanim[:to] = "transform" => "scale(1)"

And we finish by calling animate! on our Context and selecting a layer:

plt = context(200, 200) do con::Context
    group!(con, "points") do g::Group
        Gattino.points!(g, [5, 10, 15], [5, 10, 15])
    end
    Gattino.animate!(con, "points", myanim)
end

Once an animation is registered into a Context, it is there until that Context is recreated, or can be cleared manually from Context.window.extras. A defined animation can be set again when using GattinoInteractive or a similar interactive extension package for Gattino.

context plotting

plotting lines

plotting shapes

plotting other stuff

dashboards

Gattino is based on an extensible web-development framework for julia, Toolips. As a result, creating A Gattino dashboard mostly consists of composing Gattino visualizations and then serving them to a Toolips server or exporting them as HTML.

examples

styled multichart

Used By Packages

No packages found.