diff --git a/_drafts/altair_intro.md b/_drafts/altair_intro.md new file mode 100644 index 0000000..e525cc1 --- /dev/null +++ b/_drafts/altair_intro.md @@ -0,0 +1,241 @@ +--- +layout: post +title: An Introduction to Altair +twitter: true +twitter_type: summary_large_image +description: Join me in exploring Altair - a great python package for data vis +img: http://vallandingham.me/images/altair/altair_twitter.jpg +jslibs: +- 'https://cdn.jsdelivr.net/npm/vega@3' +- 'https://cdn.jsdelivr.net/npm/vega-lite@2' +- 'https://cdn.jsdelivr.net/npm/vega-embed@3' +js: 'altair.js' +zoom: false +categories: +- tutorial +--- + + + +Recently I attended a wonderful talk by [Jake VanderPlas](https://twitter.com/jakevdp) (and hosted by the always astute [Eli Bressert](https://twitter.com/astrobiased)) about a wonderful package he has been working on to make visualization in python easier. + +As a long time [ggplot2](http://ggplot2.tidyverse.org/), I'm quick to bemoan the challenges of succinctly visualizing data in python. While there are many visualization packages out there, none of them seem to approach the intuitiveness, completeness, and consistency found in R's go-to tooling. + +I'm happy to say that this new python package [Altair](https://altair-viz.github.io/index.html) might put an end to my grumblings (and yours) once and for all. + +I'd like to take you through the basics of experimenting with Altair to show a bit of what it can do. None of this is more than what you will find in the documentation, but just describes some of my experiences starting to learn how to use this tool. + +## What is Altair + +Altair provides a way to write declarative data visualization code in python by harnessing the power of [Vega](https://vega.github.io/vega/) and [Vega-Lite](https://vega.github.io/vega-lite/). Ok, so what is Vega? + +Vega is a visualization grammer (think [Grammar of Graphics](http://vita.had.co.nz/papers/layered-grammar.html) the concepts that ggplot2 is built around) that can be written as a JSON specification. Vega-lite provides most of the power of Vega in a much more compact form by relying on smart defaults and simpler encodings. + +Vega-lite provides a format to specify data, data encodings, and even [interactions](https://vega.github.io/vega/docs/event-streams/), all in a relatively simple and intuitive specification. + +In turn, Altair provides a way to generate these Vega-lite specifications using an intuitive and pythonic API. + +Intrigued? Then let's get started! + +## Jupyter Lab + +All my experimentation has been done using the newly-ready-for-use [JupyterLab](https://blog.jupyter.org/jupyterlab-is-ready-for-users-5a6f039b8906), the successor to Jupyter Notebooks. + +JupyterLab provides an updated and richer IDE-like experience for building interactive notebooks. While the new features, like an integrated text editor and terminal console, are interesting, the real exciting stuff is under the hood! + +It has been built from the ground up using a plugin-based architecture providing consistent and powerful ways to integrate new features through different types of plugins. One type of plugin, [Mime Renderer Extensions](http://jupyterlab.readthedocs.io/en/latest/developer/extension_dev.html#mime-renderer-extensions) allows for custom Javascript-based views of different filetypes. + +It is exactly this new feature that Altair leverages. It generates the Vega-Lite spec and passes it off to the built-in Vega renderer to be visualized. + +## Installation and Setup + +These instructions are based on: + +* jupyterlab v0.31.8 +* altair v2.0.0rc1 + +And might need a bit of updating as Altair is developed. First, I created a new python 3 [conda environment](https://conda.io/docs/user-guide/tasks/manage-environments.html#creating-an-environment-with-commands) to work in. Then installed jupyterlab via pip: + +```bash +pip install jupyterlab +``` + +To install the latest Altair, you need to supply the specific version at the moment: + +```bash +pip install altair==2.0.0rc1 +``` + +Finally, I had to add the `vega3` JupyterLab extension, but this step won't be necessary for future JupyterLab versions: + +```bash +jupyter labextension install @jupyterlab/vega3-extension +``` + +Now open your lab with `jupyter lab` and start a new Python 3 notebook to get started! + +## Our First Altair + +We can start with the 'Hello World' of Altair. To do this we first need to import the package. + +```python +import altair as alt +``` + +And then load some data. Handily, Altair provides some default datasets. We will start with the classic `cars` dataset. + +```python +cars = alt.load_dataset('cars') +``` + +Now let's visualize this dataset. + +```python +alt.Chart(cars).mark_point().encode( + x='Horsepower', + y='Miles_per_Gallon', + color='Origin', +) +``` +Note: The result is a Vega-Lite specification that is rendered automatically inside of JupyterLab. To show the result in this post, I will use the handle-dandy [Vega Embed](https://github.com/vega/vega-embed) to display the spec below. + +
+ +Pretty impressive, right? Let's break this code down a bit. + +First, we are creating a [Chart](https://altair-viz.github.io/user_guide/API.html#altair.Chart) instance and passing it the data that will be visualized. Next, we indicate what kind of [Mark](https://altair-viz.github.io/user_guide/marks.html) to use in the visualization. For a dot plot, we want the 'point' mark, but as we will see, there is a suite of marks to choose from. Finally, each mark has a number of possible 'channels' to encode data with. We set these in the `encoding()` method, passing in attributes (columns) of our data to use to encode `x`, `y`, and `color`. + +Again, if you are a ggplot2 user, you can start to see why I was so excited to try Altair! Marks are just another word for [Geoms](http://ggplot2.tidyverse.org/reference/#section-layer-geoms), and encodings look almost exactly like your typical `aes()` mapping! If you're not a ggplot2 user, ignore all that - and just marvel at the intuitive reading of the Altair code! + +## Same Chart, Different Data + +Now I know these test datasets are boring and dumb and boring and useless and boring. [Robert Kosara](https://eagereyes.org/blog/2018/how-to-get-excited-about-standard-datasets) is one of many voices I have heard in the recent past driving this point home. + +If there is one rational to possibly hide my laziness behind, it is that when using a boring dataset for an introductory tutorial we can focus more on the tool used to visualize it, rather than spend time looking for insights in the data. A pitiful rational, I know, so before we move on, I wanted to show how easy it can be to load up your own data and visualize with Altair. + +So we will use the wonderful [FiveThirtyEight data repository](https://github.com/fivethirtyeight/data/tree/master/candy-power-ranking) to pick out a dataset about [people's favorite candy](http://fivethirtyeight.com/features/the-ultimate-halloween-candy-power-ranking/). + +Here's how you would load and visualize this data using [pandas](https://pandas.pydata.org/) and Altair: + +```python +import pandas as pd + +candy = pd.read_csv('candy-data.csv') + +alt.Chart(candy).mark_point().encode( + x='sugarpercent', + y='winpercent', + color='hard:N', +) +``` + + + +Nearly the same command, but completely different data (we will explain that `:N` in a second). + +Most of the rest of this tutorial will be using the `cars` dataset because of my slovenliness, but keep in mind the ideas can be applied to any data. + + +## Marks Marks Everywhere + +This marks and encodings pattern starts to really shine when you want to visualize the same data differently, using different marks. Let's make a _bad_ visualization by replacing our points with bars: + +```python +alt.Chart(cars).mark_bar().encode( + x=alt.value(0), + x2='Horsepower', + y='Miles_per_Gallon', + color='Origin', +) +``` + + + +We can see the only things that have changed are the use of `mark_bar()` instead of `mark_point()`, and the additional `x2` encoding. And with those small changes, we have a whole visualization! + +Each chart can be further customized with constants provided at the mark or encoding level. For example, the opacity for each bar can be reduced by just passing in the `opacity` option to the `mark_bar()`. + +```python +alt.Chart(cars).mark_bar(opacity=0.2).encode( + x=alt.value(0), + x2='Horsepower', + y='Miles_per_Gallon', + color='Origin', +) +``` + + + + +## Faceting + +One feature i always look for in a new visualization tool or package is faceting based on a categorical variable. Many of the python packages I’ve seen require loops to accomplish this task, which I feel takes you out of the 'what' you are trying to visualize and keeps you stuck in the 'how'. Fortunately, Altair doesn’t require this cognitive break. + +You can just use the [facet()](https://altair-viz.github.io/user_guide/compound_charts.html#faceted-charts) method! + +```python +alt.Chart(cars).mark_point().encode( + x='Horsepower', + y='Miles_per_Gallon', + color='Origin', +).facet(column='Origin:N') +``` + + + +Notice the strange `:Q` trailing our data attribute. This is a special shorthand to indicate to Vega-Lite what type of data value `Origin` is. While you might think that the tool should be able to figure this out automatically, remember we are ultimate just building up a JSON specification - and so some of these nuances need to be supplied explicitly. + +## A Grammar of Interaction + +Ok. Grammar of Graphics, big deal right? You could imagine any number of tools and specifications to implement these concepts, so what makes Vega, and thus Altair any different? + +Recently, through some amazing work by the Vega team, Vega and Vega-lite now include a sort of "Grammar of Interaction". + +This means our specifications can encode not just our data visualizations, but what happens when people interact with our visualizations! + +We can get a simple pan-and-zoom interaction on any chart in Altair by simply adding the `interactive()` method: + +```python +alt.Chart(cars).mark_point().encode( + x='Horsepower', + y='Miles_per_Gallon', + color='Origin', +).facet(column='Origin:N').interactive() +``` + + + +But this is only the beginning! With Altair, we can build up much more complex interactions using [selections](https://altair-viz.github.io/user_guide/selections.html). As the Vega-Lite documentation [puts it](https://vega.github.io/vega-lite/docs/selection.html): + +> They map user input (e.g., mouse moves and clicks, touch presses, etc.) into data queries, which can subsequently be used to drive conditional encoding rules, filter data points, or determine scale domains. + +Altair comes with 3 basic selection types: + +* `selection_single()` - for interacting with a single element at a time. +* `selection_multi()` - for selecting multiple items at once through clicking or mouseover. +* `selection_interval()` - for selecting multiple items through a brushing interaction. + +Here's an example of using a `selection_interval()` to make a brush interaction that works across our previously faceted chart. + +```python +brush = alt.selection_interval() + +alt.Chart(cars).mark_point().encode( + x='Horsepower', + y='Miles_per_Gallon', + color=alt.condition(brush, 'Origin', alt.value('lightgray')), +).properties( + selection=brush +).facet(column='Origin:N') +``` + +A few interesting pieces come out of this example. First, we can see we pass this brush into our chart as its `selection` property. Next, we use it as part of a [condition](https://altair-viz.github.io/user_guide/API.html#altair.condition) - which allows us to use an if-else concept to encode the `color` channel. If the point is included in the current brush, it will be colored based on its `Origin`, else it will be `lightgray`. Try it out below by clicking and dragging the mouse. + + + +We will play a bit more with selections later on in the tutorial. + +## Summarizing Your Data + +Often times we are interested in visualizing an aggregation of the data, rather then just raw values. Altair comes equipped with a number of aggregation and binning functions that can be applied to specific encoding. + diff --git a/js/pages/altair.js b/js/pages/altair.js index 497ea63..e4a7b6b 100644 --- a/js/pages/altair.js +++ b/js/pages/altair.js @@ -4467,6 +4467,1286 @@ cars = {"values": [ } ]} +var candy = { + "values": [ + { + "competitorname": "100 Grand", + "chocolate": 1, + "fruity": 0, + "caramel": 1, + "peanutyalmondy": 0, + "nougat": 0, + "crispedricewafer": 1, + "hard": 0, + "bar": 1, + "pluribus": 0, + "sugarpercent": 0.73199999, + "pricepercent": 0.86000001, + "winpercent": 66.971725 + }, + { + "competitorname": "3 Musketeers", + "chocolate": 1, + "fruity": 0, + "caramel": 0, + "peanutyalmondy": 0, + "nougat": 1, + "crispedricewafer": 0, + "hard": 0, + "bar": 1, + "pluribus": 0, + "sugarpercent": 0.60399997, + "pricepercent": 0.51099998, + "winpercent": 67.602936 + }, + { + "competitorname": "One dime", + "chocolate": 0, + "fruity": 0, + "caramel": 0, + "peanutyalmondy": 0, + "nougat": 0, + "crispedricewafer": 0, + "hard": 0, + "bar": 0, + "pluribus": 0, + "sugarpercent": 0.011000000000000001, + "pricepercent": 0.11599999999999999, + "winpercent": 32.261086 + }, + { + "competitorname": "One quarter", + "chocolate": 0, + "fruity": 0, + "caramel": 0, + "peanutyalmondy": 0, + "nougat": 0, + "crispedricewafer": 0, + "hard": 0, + "bar": 0, + "pluribus": 0, + "sugarpercent": 0.011000000000000001, + "pricepercent": 0.51099998, + "winpercent": 46.116505 + }, + { + "competitorname": "Air Heads", + "chocolate": 0, + "fruity": 1, + "caramel": 0, + "peanutyalmondy": 0, + "nougat": 0, + "crispedricewafer": 0, + "hard": 0, + "bar": 0, + "pluribus": 0, + "sugarpercent": 0.90600002, + "pricepercent": 0.51099998, + "winpercent": 52.341465 + }, + { + "competitorname": "Almond Joy", + "chocolate": 1, + "fruity": 0, + "caramel": 0, + "peanutyalmondy": 1, + "nougat": 0, + "crispedricewafer": 0, + "hard": 0, + "bar": 1, + "pluribus": 0, + "sugarpercent": 0.465, + "pricepercent": 0.76700002, + "winpercent": 50.347546 + }, + { + "competitorname": "Baby Ruth", + "chocolate": 1, + "fruity": 0, + "caramel": 1, + "peanutyalmondy": 1, + "nougat": 1, + "crispedricewafer": 0, + "hard": 0, + "bar": 1, + "pluribus": 0, + "sugarpercent": 0.60399997, + "pricepercent": 0.76700002, + "winpercent": 56.914547 + }, + { + "competitorname": "Boston Baked Beans", + "chocolate": 0, + "fruity": 0, + "caramel": 0, + "peanutyalmondy": 1, + "nougat": 0, + "crispedricewafer": 0, + "hard": 0, + "bar": 0, + "pluribus": 1, + "sugarpercent": 0.31299999, + "pricepercent": 0.51099998, + "winpercent": 23.417824 + }, + { + "competitorname": "Candy Corn", + "chocolate": 0, + "fruity": 0, + "caramel": 0, + "peanutyalmondy": 0, + "nougat": 0, + "crispedricewafer": 0, + "hard": 0, + "bar": 0, + "pluribus": 1, + "sugarpercent": 0.90600002, + "pricepercent": 0.32499999, + "winpercent": 38.010963000000004 + }, + { + "competitorname": "Caramel Apple Pops", + "chocolate": 0, + "fruity": 1, + "caramel": 1, + "peanutyalmondy": 0, + "nougat": 0, + "crispedricewafer": 0, + "hard": 0, + "bar": 0, + "pluribus": 0, + "sugarpercent": 0.60399997, + "pricepercent": 0.32499999, + "winpercent": 34.517681 + }, + { + "competitorname": "Charleston Chew", + "chocolate": 1, + "fruity": 0, + "caramel": 0, + "peanutyalmondy": 0, + "nougat": 1, + "crispedricewafer": 0, + "hard": 0, + "bar": 1, + "pluribus": 0, + "sugarpercent": 0.60399997, + "pricepercent": 0.51099998, + "winpercent": 38.975037 + }, + { + "competitorname": "Chewey Lemonhead Fruit Mix", + "chocolate": 0, + "fruity": 1, + "caramel": 0, + "peanutyalmondy": 0, + "nougat": 0, + "crispedricewafer": 0, + "hard": 0, + "bar": 0, + "pluribus": 1, + "sugarpercent": 0.73199999, + "pricepercent": 0.51099998, + "winpercent": 36.017628 + }, + { + "competitorname": "Chiclets", + "chocolate": 0, + "fruity": 1, + "caramel": 0, + "peanutyalmondy": 0, + "nougat": 0, + "crispedricewafer": 0, + "hard": 0, + "bar": 0, + "pluribus": 1, + "sugarpercent": 0.046, + "pricepercent": 0.32499999, + "winpercent": 24.524988 + }, + { + "competitorname": "Dots", + "chocolate": 0, + "fruity": 1, + "caramel": 0, + "peanutyalmondy": 0, + "nougat": 0, + "crispedricewafer": 0, + "hard": 0, + "bar": 0, + "pluribus": 1, + "sugarpercent": 0.73199999, + "pricepercent": 0.51099998, + "winpercent": 42.272076 + }, + { + "competitorname": "Dum Dums", + "chocolate": 0, + "fruity": 1, + "caramel": 0, + "peanutyalmondy": 0, + "nougat": 0, + "crispedricewafer": 0, + "hard": 1, + "bar": 0, + "pluribus": 0, + "sugarpercent": 0.73199999, + "pricepercent": 0.034000002, + "winpercent": 39.460556 + }, + { + "competitorname": "Fruit Chews", + "chocolate": 0, + "fruity": 1, + "caramel": 0, + "peanutyalmondy": 0, + "nougat": 0, + "crispedricewafer": 0, + "hard": 0, + "bar": 0, + "pluribus": 1, + "sugarpercent": 0.127, + "pricepercent": 0.034000002, + "winpercent": 43.088924 + }, + { + "competitorname": "Fun Dip", + "chocolate": 0, + "fruity": 1, + "caramel": 0, + "peanutyalmondy": 0, + "nougat": 0, + "crispedricewafer": 0, + "hard": 1, + "bar": 0, + "pluribus": 0, + "sugarpercent": 0.73199999, + "pricepercent": 0.32499999, + "winpercent": 39.185505 + }, + { + "competitorname": "Gobstopper", + "chocolate": 0, + "fruity": 1, + "caramel": 0, + "peanutyalmondy": 0, + "nougat": 0, + "crispedricewafer": 0, + "hard": 1, + "bar": 0, + "pluribus": 1, + "sugarpercent": 0.90600002, + "pricepercent": 0.45300001, + "winpercent": 46.783348 + }, + { + "competitorname": "Haribo Gold Bears", + "chocolate": 0, + "fruity": 1, + "caramel": 0, + "peanutyalmondy": 0, + "nougat": 0, + "crispedricewafer": 0, + "hard": 0, + "bar": 0, + "pluribus": 1, + "sugarpercent": 0.465, + "pricepercent": 0.465, + "winpercent": 57.11974 + }, + { + "competitorname": "Haribo Happy Cola", + "chocolate": 0, + "fruity": 0, + "caramel": 0, + "peanutyalmondy": 0, + "nougat": 0, + "crispedricewafer": 0, + "hard": 0, + "bar": 0, + "pluribus": 1, + "sugarpercent": 0.465, + "pricepercent": 0.465, + "winpercent": 34.158958 + }, + { + "competitorname": "Haribo Sour Bears", + "chocolate": 0, + "fruity": 1, + "caramel": 0, + "peanutyalmondy": 0, + "nougat": 0, + "crispedricewafer": 0, + "hard": 0, + "bar": 0, + "pluribus": 1, + "sugarpercent": 0.465, + "pricepercent": 0.465, + "winpercent": 51.41243 + }, + { + "competitorname": "Haribo Twin Snakes", + "chocolate": 0, + "fruity": 1, + "caramel": 0, + "peanutyalmondy": 0, + "nougat": 0, + "crispedricewafer": 0, + "hard": 0, + "bar": 0, + "pluribus": 1, + "sugarpercent": 0.465, + "pricepercent": 0.465, + "winpercent": 42.178771999999995 + }, + { + "competitorname": "HersheyÕs Kisses", + "chocolate": 1, + "fruity": 0, + "caramel": 0, + "peanutyalmondy": 0, + "nougat": 0, + "crispedricewafer": 0, + "hard": 0, + "bar": 0, + "pluribus": 1, + "sugarpercent": 0.127, + "pricepercent": 0.093000002, + "winpercent": 55.375454000000005 + }, + { + "competitorname": "HersheyÕs Krackel", + "chocolate": 1, + "fruity": 0, + "caramel": 0, + "peanutyalmondy": 0, + "nougat": 0, + "crispedricewafer": 1, + "hard": 0, + "bar": 1, + "pluribus": 0, + "sugarpercent": 0.43000001, + "pricepercent": 0.91799998, + "winpercent": 62.28448100000001 + }, + { + "competitorname": "HersheyÕs Milk Chocolate", + "chocolate": 1, + "fruity": 0, + "caramel": 0, + "peanutyalmondy": 0, + "nougat": 0, + "crispedricewafer": 0, + "hard": 0, + "bar": 1, + "pluribus": 0, + "sugarpercent": 0.43000001, + "pricepercent": 0.91799998, + "winpercent": 56.490501 + }, + { + "competitorname": "HersheyÕs Special Dark", + "chocolate": 1, + "fruity": 0, + "caramel": 0, + "peanutyalmondy": 0, + "nougat": 0, + "crispedricewafer": 0, + "hard": 0, + "bar": 1, + "pluribus": 0, + "sugarpercent": 0.43000001, + "pricepercent": 0.91799998, + "winpercent": 59.236121999999995 + }, + { + "competitorname": "Jawbusters", + "chocolate": 0, + "fruity": 1, + "caramel": 0, + "peanutyalmondy": 0, + "nougat": 0, + "crispedricewafer": 0, + "hard": 1, + "bar": 0, + "pluribus": 1, + "sugarpercent": 0.093000002, + "pricepercent": 0.51099998, + "winpercent": 28.127439000000003 + }, + { + "competitorname": "Junior Mints", + "chocolate": 1, + "fruity": 0, + "caramel": 0, + "peanutyalmondy": 0, + "nougat": 0, + "crispedricewafer": 0, + "hard": 0, + "bar": 0, + "pluribus": 1, + "sugarpercent": 0.19699999999999998, + "pricepercent": 0.51099998, + "winpercent": 57.21925 + }, + { + "competitorname": "Kit Kat", + "chocolate": 1, + "fruity": 0, + "caramel": 0, + "peanutyalmondy": 0, + "nougat": 0, + "crispedricewafer": 1, + "hard": 0, + "bar": 1, + "pluribus": 0, + "sugarpercent": 0.31299999, + "pricepercent": 0.51099998, + "winpercent": 76.7686 + }, + { + "competitorname": "Laffy Taffy", + "chocolate": 0, + "fruity": 1, + "caramel": 0, + "peanutyalmondy": 0, + "nougat": 0, + "crispedricewafer": 0, + "hard": 0, + "bar": 0, + "pluribus": 0, + "sugarpercent": 0.22, + "pricepercent": 0.11599999999999999, + "winpercent": 41.389557 + }, + { + "competitorname": "Lemonhead", + "chocolate": 0, + "fruity": 1, + "caramel": 0, + "peanutyalmondy": 0, + "nougat": 0, + "crispedricewafer": 0, + "hard": 1, + "bar": 0, + "pluribus": 0, + "sugarpercent": 0.046, + "pricepercent": 0.10400000000000001, + "winpercent": 39.141056 + }, + { + "competitorname": "Lifesavers big ring gummies", + "chocolate": 0, + "fruity": 1, + "caramel": 0, + "peanutyalmondy": 0, + "nougat": 0, + "crispedricewafer": 0, + "hard": 0, + "bar": 0, + "pluribus": 0, + "sugarpercent": 0.26699999, + "pricepercent": 0.27900001, + "winpercent": 52.911392000000006 + }, + { + "competitorname": "Peanut butter M&MÕs", + "chocolate": 1, + "fruity": 0, + "caramel": 0, + "peanutyalmondy": 1, + "nougat": 0, + "crispedricewafer": 0, + "hard": 0, + "bar": 0, + "pluribus": 1, + "sugarpercent": 0.82499999, + "pricepercent": 0.65100002, + "winpercent": 71.46505 + }, + { + "competitorname": "M&MÕs", + "chocolate": 1, + "fruity": 0, + "caramel": 0, + "peanutyalmondy": 0, + "nougat": 0, + "crispedricewafer": 0, + "hard": 0, + "bar": 0, + "pluribus": 1, + "sugarpercent": 0.82499999, + "pricepercent": 0.65100002, + "winpercent": 66.574585 + }, + { + "competitorname": "Mike & Ike", + "chocolate": 0, + "fruity": 1, + "caramel": 0, + "peanutyalmondy": 0, + "nougat": 0, + "crispedricewafer": 0, + "hard": 0, + "bar": 0, + "pluribus": 1, + "sugarpercent": 0.87199998, + "pricepercent": 0.32499999, + "winpercent": 46.411716 + }, + { + "competitorname": "Milk Duds", + "chocolate": 1, + "fruity": 0, + "caramel": 1, + "peanutyalmondy": 0, + "nougat": 0, + "crispedricewafer": 0, + "hard": 0, + "bar": 0, + "pluribus": 1, + "sugarpercent": 0.30199999, + "pricepercent": 0.51099998, + "winpercent": 55.064071999999996 + }, + { + "competitorname": "Milky Way", + "chocolate": 1, + "fruity": 0, + "caramel": 1, + "peanutyalmondy": 0, + "nougat": 1, + "crispedricewafer": 0, + "hard": 0, + "bar": 1, + "pluribus": 0, + "sugarpercent": 0.60399997, + "pricepercent": 0.65100002, + "winpercent": 73.099556 + }, + { + "competitorname": "Milky Way Midnight", + "chocolate": 1, + "fruity": 0, + "caramel": 1, + "peanutyalmondy": 0, + "nougat": 1, + "crispedricewafer": 0, + "hard": 0, + "bar": 1, + "pluribus": 0, + "sugarpercent": 0.73199999, + "pricepercent": 0.44100001, + "winpercent": 60.800701000000004 + }, + { + "competitorname": "Milky Way Simply Caramel", + "chocolate": 1, + "fruity": 0, + "caramel": 1, + "peanutyalmondy": 0, + "nougat": 0, + "crispedricewafer": 0, + "hard": 0, + "bar": 1, + "pluribus": 0, + "sugarpercent": 0.96499997, + "pricepercent": 0.86000001, + "winpercent": 64.35334 + }, + { + "competitorname": "Mounds", + "chocolate": 1, + "fruity": 0, + "caramel": 0, + "peanutyalmondy": 0, + "nougat": 0, + "crispedricewafer": 0, + "hard": 0, + "bar": 1, + "pluribus": 0, + "sugarpercent": 0.31299999, + "pricepercent": 0.86000001, + "winpercent": 47.829754 + }, + { + "competitorname": "Mr Good Bar", + "chocolate": 1, + "fruity": 0, + "caramel": 0, + "peanutyalmondy": 1, + "nougat": 0, + "crispedricewafer": 0, + "hard": 0, + "bar": 1, + "pluribus": 0, + "sugarpercent": 0.31299999, + "pricepercent": 0.91799998, + "winpercent": 54.526451 + }, + { + "competitorname": "Nerds", + "chocolate": 0, + "fruity": 1, + "caramel": 0, + "peanutyalmondy": 0, + "nougat": 0, + "crispedricewafer": 0, + "hard": 1, + "bar": 0, + "pluribus": 1, + "sugarpercent": 0.84799999, + "pricepercent": 0.32499999, + "winpercent": 55.354046 + }, + { + "competitorname": "Nestle Butterfinger", + "chocolate": 1, + "fruity": 0, + "caramel": 0, + "peanutyalmondy": 1, + "nougat": 0, + "crispedricewafer": 0, + "hard": 0, + "bar": 1, + "pluribus": 0, + "sugarpercent": 0.60399997, + "pricepercent": 0.76700002, + "winpercent": 70.735641 + }, + { + "competitorname": "Nestle Crunch", + "chocolate": 1, + "fruity": 0, + "caramel": 0, + "peanutyalmondy": 0, + "nougat": 0, + "crispedricewafer": 1, + "hard": 0, + "bar": 1, + "pluribus": 0, + "sugarpercent": 0.31299999, + "pricepercent": 0.76700002, + "winpercent": 66.47068 + }, + { + "competitorname": "Nik L Nip", + "chocolate": 0, + "fruity": 1, + "caramel": 0, + "peanutyalmondy": 0, + "nougat": 0, + "crispedricewafer": 0, + "hard": 0, + "bar": 0, + "pluribus": 1, + "sugarpercent": 0.19699999999999998, + "pricepercent": 0.97600001, + "winpercent": 22.445341 + }, + { + "competitorname": "Now & Later", + "chocolate": 0, + "fruity": 1, + "caramel": 0, + "peanutyalmondy": 0, + "nougat": 0, + "crispedricewafer": 0, + "hard": 0, + "bar": 0, + "pluribus": 1, + "sugarpercent": 0.22, + "pricepercent": 0.32499999, + "winpercent": 39.4468 + }, + { + "competitorname": "Payday", + "chocolate": 0, + "fruity": 0, + "caramel": 0, + "peanutyalmondy": 1, + "nougat": 1, + "crispedricewafer": 0, + "hard": 0, + "bar": 1, + "pluribus": 0, + "sugarpercent": 0.465, + "pricepercent": 0.76700002, + "winpercent": 46.296597 + }, + { + "competitorname": "Peanut M&Ms", + "chocolate": 1, + "fruity": 0, + "caramel": 0, + "peanutyalmondy": 1, + "nougat": 0, + "crispedricewafer": 0, + "hard": 0, + "bar": 0, + "pluribus": 1, + "sugarpercent": 0.59299999, + "pricepercent": 0.65100002, + "winpercent": 69.483788 + }, + { + "competitorname": "Pixie Sticks", + "chocolate": 0, + "fruity": 0, + "caramel": 0, + "peanutyalmondy": 0, + "nougat": 0, + "crispedricewafer": 0, + "hard": 0, + "bar": 0, + "pluribus": 1, + "sugarpercent": 0.093000002, + "pricepercent": 0.023, + "winpercent": 37.722336 + }, + { + "competitorname": "Pop Rocks", + "chocolate": 0, + "fruity": 1, + "caramel": 0, + "peanutyalmondy": 0, + "nougat": 0, + "crispedricewafer": 0, + "hard": 1, + "bar": 0, + "pluribus": 1, + "sugarpercent": 0.60399997, + "pricepercent": 0.83700001, + "winpercent": 41.265511 + }, + { + "competitorname": "Red vines", + "chocolate": 0, + "fruity": 1, + "caramel": 0, + "peanutyalmondy": 0, + "nougat": 0, + "crispedricewafer": 0, + "hard": 0, + "bar": 0, + "pluribus": 1, + "sugarpercent": 0.58099997, + "pricepercent": 0.11599999999999999, + "winpercent": 37.348521999999996 + }, + { + "competitorname": "ReeseÕs Miniatures", + "chocolate": 1, + "fruity": 0, + "caramel": 0, + "peanutyalmondy": 1, + "nougat": 0, + "crispedricewafer": 0, + "hard": 0, + "bar": 0, + "pluribus": 0, + "sugarpercent": 0.034000002, + "pricepercent": 0.27900001, + "winpercent": 81.86625699999999 + }, + { + "competitorname": "ReeseÕs Peanut Butter cup", + "chocolate": 1, + "fruity": 0, + "caramel": 0, + "peanutyalmondy": 1, + "nougat": 0, + "crispedricewafer": 0, + "hard": 0, + "bar": 0, + "pluribus": 0, + "sugarpercent": 0.72000003, + "pricepercent": 0.65100002, + "winpercent": 84.18029 + }, + { + "competitorname": "ReeseÕs pieces", + "chocolate": 1, + "fruity": 0, + "caramel": 0, + "peanutyalmondy": 1, + "nougat": 0, + "crispedricewafer": 0, + "hard": 0, + "bar": 0, + "pluribus": 1, + "sugarpercent": 0.40599999, + "pricepercent": 0.65100002, + "winpercent": 73.43499 + }, + { + "competitorname": "ReeseÕs stuffed with pieces", + "chocolate": 1, + "fruity": 0, + "caramel": 0, + "peanutyalmondy": 1, + "nougat": 0, + "crispedricewafer": 0, + "hard": 0, + "bar": 0, + "pluribus": 0, + "sugarpercent": 0.98799998, + "pricepercent": 0.65100002, + "winpercent": 72.887901 + }, + { + "competitorname": "Ring pop", + "chocolate": 0, + "fruity": 1, + "caramel": 0, + "peanutyalmondy": 0, + "nougat": 0, + "crispedricewafer": 0, + "hard": 1, + "bar": 0, + "pluribus": 0, + "sugarpercent": 0.73199999, + "pricepercent": 0.96499997, + "winpercent": 35.290756 + }, + { + "competitorname": "Rolo", + "chocolate": 1, + "fruity": 0, + "caramel": 1, + "peanutyalmondy": 0, + "nougat": 0, + "crispedricewafer": 0, + "hard": 0, + "bar": 0, + "pluribus": 1, + "sugarpercent": 0.86000001, + "pricepercent": 0.86000001, + "winpercent": 65.716286 + }, + { + "competitorname": "Root Beer Barrels", + "chocolate": 0, + "fruity": 0, + "caramel": 0, + "peanutyalmondy": 0, + "nougat": 0, + "crispedricewafer": 0, + "hard": 1, + "bar": 0, + "pluribus": 1, + "sugarpercent": 0.73199999, + "pricepercent": 0.06899999799999999, + "winpercent": 29.703691 + }, + { + "competitorname": "Runts", + "chocolate": 0, + "fruity": 1, + "caramel": 0, + "peanutyalmondy": 0, + "nougat": 0, + "crispedricewafer": 0, + "hard": 1, + "bar": 0, + "pluribus": 1, + "sugarpercent": 0.87199998, + "pricepercent": 0.27900001, + "winpercent": 42.849144 + }, + { + "competitorname": "Sixlets", + "chocolate": 1, + "fruity": 0, + "caramel": 0, + "peanutyalmondy": 0, + "nougat": 0, + "crispedricewafer": 0, + "hard": 0, + "bar": 0, + "pluribus": 1, + "sugarpercent": 0.22, + "pricepercent": 0.081, + "winpercent": 34.722 + }, + { + "competitorname": "Skittles original", + "chocolate": 0, + "fruity": 1, + "caramel": 0, + "peanutyalmondy": 0, + "nougat": 0, + "crispedricewafer": 0, + "hard": 0, + "bar": 0, + "pluribus": 1, + "sugarpercent": 0.94099998, + "pricepercent": 0.22, + "winpercent": 63.08514 + }, + { + "competitorname": "Skittles wildberry", + "chocolate": 0, + "fruity": 1, + "caramel": 0, + "peanutyalmondy": 0, + "nougat": 0, + "crispedricewafer": 0, + "hard": 0, + "bar": 0, + "pluribus": 1, + "sugarpercent": 0.94099998, + "pricepercent": 0.22, + "winpercent": 55.103694999999995 + }, + { + "competitorname": "Nestle Smarties", + "chocolate": 1, + "fruity": 0, + "caramel": 0, + "peanutyalmondy": 0, + "nougat": 0, + "crispedricewafer": 0, + "hard": 0, + "bar": 0, + "pluribus": 1, + "sugarpercent": 0.26699999, + "pricepercent": 0.97600001, + "winpercent": 37.887188 + }, + { + "competitorname": "Smarties candy", + "chocolate": 0, + "fruity": 1, + "caramel": 0, + "peanutyalmondy": 0, + "nougat": 0, + "crispedricewafer": 0, + "hard": 1, + "bar": 0, + "pluribus": 1, + "sugarpercent": 0.26699999, + "pricepercent": 0.11599999999999999, + "winpercent": 45.995827 + }, + { + "competitorname": "Snickers", + "chocolate": 1, + "fruity": 0, + "caramel": 1, + "peanutyalmondy": 1, + "nougat": 1, + "crispedricewafer": 0, + "hard": 0, + "bar": 1, + "pluribus": 0, + "sugarpercent": 0.546, + "pricepercent": 0.65100002, + "winpercent": 76.67378199999999 + }, + { + "competitorname": "Snickers Crisper", + "chocolate": 1, + "fruity": 0, + "caramel": 1, + "peanutyalmondy": 1, + "nougat": 0, + "crispedricewafer": 1, + "hard": 0, + "bar": 1, + "pluribus": 0, + "sugarpercent": 0.60399997, + "pricepercent": 0.65100002, + "winpercent": 59.529251 + }, + { + "competitorname": "Sour Patch Kids", + "chocolate": 0, + "fruity": 1, + "caramel": 0, + "peanutyalmondy": 0, + "nougat": 0, + "crispedricewafer": 0, + "hard": 0, + "bar": 0, + "pluribus": 1, + "sugarpercent": 0.06899999799999999, + "pricepercent": 0.11599999999999999, + "winpercent": 59.863997999999995 + }, + { + "competitorname": "Sour Patch Tricksters", + "chocolate": 0, + "fruity": 1, + "caramel": 0, + "peanutyalmondy": 0, + "nougat": 0, + "crispedricewafer": 0, + "hard": 0, + "bar": 0, + "pluribus": 1, + "sugarpercent": 0.06899999799999999, + "pricepercent": 0.11599999999999999, + "winpercent": 52.825947 + }, + { + "competitorname": "Starburst", + "chocolate": 0, + "fruity": 1, + "caramel": 0, + "peanutyalmondy": 0, + "nougat": 0, + "crispedricewafer": 0, + "hard": 0, + "bar": 0, + "pluribus": 1, + "sugarpercent": 0.15099999, + "pricepercent": 0.22, + "winpercent": 67.037628 + }, + { + "competitorname": "Strawberry bon bons", + "chocolate": 0, + "fruity": 1, + "caramel": 0, + "peanutyalmondy": 0, + "nougat": 0, + "crispedricewafer": 0, + "hard": 1, + "bar": 0, + "pluribus": 1, + "sugarpercent": 0.56900001, + "pricepercent": 0.057999998, + "winpercent": 34.578990999999995 + }, + { + "competitorname": "Sugar Babies", + "chocolate": 0, + "fruity": 0, + "caramel": 1, + "peanutyalmondy": 0, + "nougat": 0, + "crispedricewafer": 0, + "hard": 0, + "bar": 0, + "pluribus": 1, + "sugarpercent": 0.96499997, + "pricepercent": 0.76700002, + "winpercent": 33.43755 + }, + { + "competitorname": "Sugar Daddy", + "chocolate": 0, + "fruity": 0, + "caramel": 1, + "peanutyalmondy": 0, + "nougat": 0, + "crispedricewafer": 0, + "hard": 0, + "bar": 0, + "pluribus": 0, + "sugarpercent": 0.41800001, + "pricepercent": 0.32499999, + "winpercent": 32.230995 + }, + { + "competitorname": "Super Bubble", + "chocolate": 0, + "fruity": 1, + "caramel": 0, + "peanutyalmondy": 0, + "nougat": 0, + "crispedricewafer": 0, + "hard": 0, + "bar": 0, + "pluribus": 0, + "sugarpercent": 0.162, + "pricepercent": 0.11599999999999999, + "winpercent": 27.303865000000002 + }, + { + "competitorname": "Swedish Fish", + "chocolate": 0, + "fruity": 1, + "caramel": 0, + "peanutyalmondy": 0, + "nougat": 0, + "crispedricewafer": 0, + "hard": 0, + "bar": 0, + "pluribus": 1, + "sugarpercent": 0.60399997, + "pricepercent": 0.755, + "winpercent": 54.861111 + }, + { + "competitorname": "Tootsie Pop", + "chocolate": 1, + "fruity": 1, + "caramel": 0, + "peanutyalmondy": 0, + "nougat": 0, + "crispedricewafer": 0, + "hard": 1, + "bar": 0, + "pluribus": 0, + "sugarpercent": 0.60399997, + "pricepercent": 0.32499999, + "winpercent": 48.982651000000004 + }, + { + "competitorname": "Tootsie Roll Juniors", + "chocolate": 1, + "fruity": 0, + "caramel": 0, + "peanutyalmondy": 0, + "nougat": 0, + "crispedricewafer": 0, + "hard": 0, + "bar": 0, + "pluribus": 0, + "sugarpercent": 0.31299999, + "pricepercent": 0.51099998, + "winpercent": 43.068897 + }, + { + "competitorname": "Tootsie Roll Midgies", + "chocolate": 1, + "fruity": 0, + "caramel": 0, + "peanutyalmondy": 0, + "nougat": 0, + "crispedricewafer": 0, + "hard": 0, + "bar": 0, + "pluribus": 1, + "sugarpercent": 0.17399999, + "pricepercent": 0.011000000000000001, + "winpercent": 45.736748 + }, + { + "competitorname": "Tootsie Roll Snack Bars", + "chocolate": 1, + "fruity": 0, + "caramel": 0, + "peanutyalmondy": 0, + "nougat": 0, + "crispedricewafer": 0, + "hard": 0, + "bar": 1, + "pluribus": 0, + "sugarpercent": 0.465, + "pricepercent": 0.32499999, + "winpercent": 49.653503 + }, + { + "competitorname": "Trolli Sour Bites", + "chocolate": 0, + "fruity": 1, + "caramel": 0, + "peanutyalmondy": 0, + "nougat": 0, + "crispedricewafer": 0, + "hard": 0, + "bar": 0, + "pluribus": 1, + "sugarpercent": 0.31299999, + "pricepercent": 0.255, + "winpercent": 47.173229 + }, + { + "competitorname": "Twix", + "chocolate": 1, + "fruity": 0, + "caramel": 1, + "peanutyalmondy": 0, + "nougat": 0, + "crispedricewafer": 1, + "hard": 0, + "bar": 1, + "pluribus": 0, + "sugarpercent": 0.546, + "pricepercent": 0.90600002, + "winpercent": 81.642914 + }, + { + "competitorname": "Twizzlers", + "chocolate": 0, + "fruity": 1, + "caramel": 0, + "peanutyalmondy": 0, + "nougat": 0, + "crispedricewafer": 0, + "hard": 0, + "bar": 0, + "pluribus": 0, + "sugarpercent": 0.22, + "pricepercent": 0.11599999999999999, + "winpercent": 45.466282 + }, + { + "competitorname": "Warheads", + "chocolate": 0, + "fruity": 1, + "caramel": 0, + "peanutyalmondy": 0, + "nougat": 0, + "crispedricewafer": 0, + "hard": 1, + "bar": 0, + "pluribus": 0, + "sugarpercent": 0.093000002, + "pricepercent": 0.11599999999999999, + "winpercent": 39.011897999999995 + }, + { + "competitorname": "WelchÕs Fruit Snacks", + "chocolate": 0, + "fruity": 1, + "caramel": 0, + "peanutyalmondy": 0, + "nougat": 0, + "crispedricewafer": 0, + "hard": 0, + "bar": 0, + "pluribus": 1, + "sugarpercent": 0.31299999, + "pricepercent": 0.31299999, + "winpercent": 44.375519 + }, + { + "competitorname": "WertherÕs Original Caramel", + "chocolate": 0, + "fruity": 0, + "caramel": 1, + "peanutyalmondy": 0, + "nougat": 0, + "crispedricewafer": 0, + "hard": 1, + "bar": 0, + "pluribus": 0, + "sugarpercent": 0.18600000000000003, + "pricepercent": 0.26699999, + "winpercent": 41.904308 + }, + { + "competitorname": "Whoppers", + "chocolate": 1, + "fruity": 0, + "caramel": 0, + "peanutyalmondy": 0, + "nougat": 0, + "crispedricewafer": 1, + "hard": 0, + "bar": 0, + "pluribus": 1, + "sugarpercent": 0.87199998, + "pricepercent": 0.84799999, + "winpercent": 49.524113 + } + ] +} + var spec1 = { "config": { "view": { @@ -4493,6 +5773,8 @@ var spec1 = { "$schema": "https://vega.github.io/schema/vega-lite/v2.json" } +vegaEmbed('#spec1', spec1, {actions: false}) + var spec2 = { "$schema": "https://vega.github.io/schema/vega-lite/v2.json", "config": { @@ -4557,4 +5839,182 @@ var spec3 = { } } -vegaEmbed('#spec3', spec3, {actions: false}) \ No newline at end of file + +vegaEmbed('#spec3', spec3, {actions: false}) + +var spec4 = { + "config": { + "view": { + "width": 400, + "height": 300 + } + }, + "data": candy, + "mark": "point", + "encoding": { + "color": { + "type": "nominal", + "field": "hard" + }, + "x": { + "type": "quantitative", + "field": "sugarpercent" + }, + "y": { + "type": "quantitative", + "field": "winpercent" + } + }, + "$schema": "https://vega.github.io/schema/vega-lite/v2.json" +} + +vegaEmbed('#spec4', spec4, {actions: false}) + +var spec5 = { + "$schema": "https://vega.github.io/schema/vega-lite/v2.json", + "config": { + "view": { + "height": 300, + "width": 240 + } + }, + "data": cars, + "facet": { + "column": { + "field": "Origin", + "type": "nominal" + } + }, + "spec": { + "encoding": { + "color": { + "field": "Origin", + "type": "nominal" + }, + "x": { + "field": "Horsepower", + "type": "quantitative" + }, + "y": { + "field": "Miles_per_Gallon", + "type": "quantitative" + }, + }, + "mark": "point" + } +} + +vegaEmbed('#spec5', spec5, {actions: false}) + +var spec6 = { + "$schema": "https://vega.github.io/schema/vega-lite/v2.json", + "config": { + "view": { + "height": 300, + "width": 240 + } + }, + "data": cars, + "facet": { + "column": { + "field": "Origin", + "type": "nominal" + } + }, + "spec": { + "encoding": { + "color": { + "field": "Origin", + "type": "nominal" + }, + "x": { + "field": "Horsepower", + "type": "quantitative" + }, + "y": { + "field": "Miles_per_Gallon", + "type": "quantitative" + } + }, + "mark": "point", + "selection": { + "selector001": { + "bind": "scales", + "encodings": [ + "x", + "y" + ], + "mark": { + "fill": "#333", + "fillOpacity": 0.125, + "stroke": "white" + }, + "on": "[mousedown, window:mouseup] > window:mousemove!", + "resolve": "global", + "translate": "[mousedown, window:mouseup] > window:mousemove!", + "type": "interval", + "zoom": "wheel!" + } + } + } +} + +vegaEmbed('#spec6', spec6, {actions: false}) + +var spec7 = { + "$schema": "https://vega.github.io/schema/vega-lite/v2.json", + "config": { + "view": { + "height": 300, + "width": 240 + } + }, + "data": cars, + "facet": { + "column": { + "field": "Origin", + "type": "nominal" + } + }, + "spec": { + "encoding": { + "color": { + "condition": { + "field": "Origin", + "selection": "selector002", + "type": "nominal" + }, + "value": "lightgray" + }, + "x": { + "field": "Horsepower", + "type": "quantitative" + }, + "y": { + "field": "Miles_per_Gallon", + "type": "quantitative" + } + }, + "mark": "point", + "selection": { + "selector002": { + "encodings": [ + "x", + "y" + ], + "mark": { + "fill": "#333", + "fillOpacity": 0.125, + "stroke": "white" + }, + "on": "[mousedown, window:mouseup] > window:mousemove!", + "resolve": "global", + "translate": "[mousedown, window:mouseup] > window:mousemove!", + "type": "interval", + "zoom": "wheel!" + } + } + } +} + +vegaEmbed('#spec7', spec7, {actions: false}) \ No newline at end of file