Skip to content

Commit

Permalink
finsh function until expand_node_view
Browse files Browse the repository at this point in the history
  • Loading branch information
YunchaoLiu committed Sep 8, 2018
1 parent 90cbf9e commit 8cbb141
Show file tree
Hide file tree
Showing 12 changed files with 457 additions and 6 deletions.
Binary file added assignment0.zip
Binary file not shown.
Binary file added assignment0/assignment0.jpg
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
10 changes: 5 additions & 5 deletions assignment0/index.html
Original file line number Diff line number Diff line change
Expand Up @@ -10,15 +10,15 @@

// TODO: fill in the a's!
var q1 = 'What would you like to get out of this course?'
var a1 = ''
var a1 = 'Learn how to visulize data so I can analyse data better.'
var q2 = 'What are your research interests?'
var a2 = ''
var a2 = 'Protein folding'
var q3 = 'What is your experience with web development?'
var a3 = ''
var a3 = 'I have very limited experience'
var q4 = 'What is your experience with D3?'
var a4 = ''
var a4 = 'none'
var q5 = 'Have you used visualization tools and libraries before, such as Tableau, matplotlib, ggplot2, etc..?'
var a5 = ''
var a5 = 'Tableau, but for a very short time'
var qs_and_as = [{'statement_type':'question','value':q1},{'statement_type':'answer','value':a1},
{'statement_type':'question','value':q2},{'statement_type':'answer','value':a2},
{'statement_type':'question','value':q3},{'statement_type':'answer','value':a3},
Expand Down
Binary file added assignment1.zip
Binary file not shown.
2 changes: 1 addition & 1 deletion assignment1/README.md
Original file line number Diff line number Diff line change
Expand Up @@ -30,5 +30,5 @@ Zip up the `assignment1` directory with `part1.html` and `part2.html` completed,
### Part 2 - 65%

* 30%: correcty join data and assign element attributes for the background and `bar` data
* 10%: correcty join data and assign element attributes for the `box` data and `main_circles` data
* 15%: correcty join data and assign element attributes for the `box` data and `main_circles` data
* 20%: correcty join data and assign attributes for the `eight_circles` data
66 changes: 66 additions & 0 deletions assignment1/part2.html
Original file line number Diff line number Diff line change
@@ -1,3 +1,4 @@
<<<<<<< HEAD
<!DOCTYPE html>
<html lang='en'>
<head>
Expand Down Expand Up @@ -81,3 +82,68 @@

</body>
</html>
=======
<!DOCTYPE html>
<html lang='en'>
<head>
<meta charset='utf-8'>
<title>Assignment 1 - D3 Workout Pt. 2</title>
<script type='text/javascript' src='../d3.js'></script>
</head>
<body>

<script type='text/javascript'>
{
var bg_color = '#ed8369';
var black_color = '#000000';
var bar_color_1 = '#ffecbf';
var bar_color_2 = '#e2ca9e';

// each bar is: x,y,width -> the baseline should be at the very bottom
var bars = [
[29,613,60-29],
[102,613,139-102],
[197,619,242-197],
[292,650,348-292],
[398,702,445-398],
[503,722,537-503],
[570,793,624-570],
[679,809,726-679],
[811,814,840-811],
[905,822,933-905],
[1016,787,1074-1016],
[1123,712,1181-1123]
];

var box = [[24,444,143-24,579-444]]; // x,y,width,height

// x,y,radius,fill
var main_circles = [
[702.6,769,20.4,black_color],
[702.6,769,8.4,bg_color],
];

// grouped according to left, right: 'position' is where the circles are centered, children formated as [radius,color]
var eight_circles = [
{'position':[70,507], 'children':[[23,bg_color],[11,black_color]]},
{'position':[104.5,507], 'children':[[21,bg_color],[8.4,black_color]]}
];

d3.select('body').append('svg').attr('width', '1200').attr('height', '1200');

// TODO: create a `rect` with fill `bg_color`, and whose shape matches that of the svg element

// TODO: create `rect`s from the `bars` data, with fill `bar_color_1` when the i'th bar is even and `bar_color_2` if the i'th bar is odd

// TODO: create `rect` from the `box` data, with fill `black_color`

// TODO: create `circle`s from `main_circles`, using colors in the data for fill

// TODO: first create group elements from the outer array, using the position field to define the transformation
// then, create a nested selection using the children field to propagate data
}
</script>

</body>
</html>
>>>>>>> ddccec34d0d8095392dc0a7fe23f1be3d23699a8
Binary file added assignment2/.interactive_time_series.js.swp
Binary file not shown.
126 changes: 126 additions & 0 deletions assignment2/README.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,126 @@
# Assignment 2

In this assignment you will be combining all of your knowledge in D3 for interactively exploring multiple time series data that is hierarchically organized. The main javascript code that you will be starting from for this assignment can be found in [http://localhost:8000/assignment2/interactive_time_series.js](http://localhost:8000/assignment2/interactive_time_series.js), and the index is [http://localhost:8000/assignment2/index.html](http://localhost:8000/assignment2/index.html).

## Data

There are two time series datasets that you will be provided for experimentation. Both are formatted in the same exact way (minus date formats, which are accounted for). Your code should be able to visualize both with no modifications (other than switching the dataset, see `is_music_time_series` variable in index).

### Population Counts in Japan

This dataset contains a set of time series of population counts for the country of Japan. More specifically, each time series is categorized by the following hierarchy levels:

* country (the root node: Japan)
* three of Japan's main island regions
* within each island, a number of prefectures (geographic regions)

Thus, a single time series contains population counts for a particular prefecture, this prefecture belongs to a particular island, and the island belongs to Japan.

[Click here to download this dataset](https://vanderbilt.box.com/s/y52ufqgx7ptxcb0zh0e84lya4b4l0jkw), and place it in the assignment2 directory.

### Music Listening Counts from last.fm

This dataset contains a set of time series of music tracks that are "scrobbled" (listened to) by the music community of the website last.fm. The tracks are broken down by genre, consisting of the following hierarchy levels:

* all musical genres (the root node)
* five general musical genres
* within each genre, a set of more specific subgenres

Thus, a single time series contains listening counts for a particular subgenre, this subgenre belongs to a particular genre, and the genre belongs to the root (all genres of music).

[Click here to download this dataset](https://vanderbilt.box.com/s/ktixa1e5jn5avnig8vu1wk3vpu3822hf), and place it in the assignment2 directory.

### Data Format

Each dataset is hierarchically organized via JSON, and upon loaded through D3 it is an object with a tree data structure, with each node containing the following properties:

* `counts`: this property only exists at the leaf nodes, and is an array that consists of objects containing properties:
* `date`: the date at which this count exists
* `count`: the count for this date
* `name`: the name of this node
* `children`: an array of nodes that are children of `this` node, empty for leaf nodes

Furthermore, the dates are ordered, so there is no need to sort `counts`.

### Aggregation

Although we could simply visualize all of the time series at once, this will result in clutter, and will make it challenging to compare time series. Thus, you will first be responsible for **aggregating** the time series: namely, you will compute a time series for each non-leaf node that aggregates the time series of its children. The aggregated time series should be stored as an array in new property `counts` (similar to leaf nodes), where each entry in the array has the same structure as `counts` within the leaf nodes (an object containing properties `date` and `count`). You should be able to support two different types of aggregations:

* mean
* sum

Note that you are only to aggregate on `count`, `date` is identical across time series.

We will then want to visualize this hierarchy.

## Visual Encoding

You will use line charts to visualize the time series data. In particular, you will visualize particular types of **node subsets** of the hierarchy, where each node in the hierarchy contains a time series. More specifically, a valid subset is one that:

* **Covers all children**: all leaves in the tree can be reached by a path that originates in one of the nodes in the subset (a leaf node in the subset is considered to be already "reached").
* **Excludes ancestors**: two nodes in which one is an ancestor of another cannot belong in the subset.

For any subset that satisfies these conditions, you will plot all of their corresponding time series in one graph. To realize this, you will need:

* `d3.scales` for the x and y axes, as well as `d3.line` to plot each time series.
* `d3.axis` to show the axes
* labels to show the axes - in particular, for each time series, you will place a text label on the right side of the graph, whose y-coordinate is assigned that of the time series' (mapped via scale) value at the last date, so that the text label is adjacent to the time series.

Moreover, we will want to process any subset of nodes, which means that we must be able to match nodes when we perform data joins. Thus, **keys** should be used when performing data joins, and you may use the `name` property of each node for this purpose.

Color has also been pre-defined for the nodes of the tree in the `color` property. You are to assign color, stored in the nodes, to the path elements, as well as the text elements.

## Interactivity

The time series are organized in this manner to enable interactivity. To this end, you will need to support the following operations:

* **Expand**: when a user performs a `click` on a time series, it should be replaced by its **direct children**. That is, the current time series is removed, and its children time series are added.
* **Collapse**: when a user performs a `Shift+click` on a time series, it should be replaced by its **parent**. Specifically, the parent time series is added and any node that is a descendant of this parent is removed.

You will need to do two main operations here:

* For a given time series that the user clicked on, you must determine what nodes in the tree to visualize, depending on whether they expand or collapse.
* Given this array of nodes, perform the standard update pattern that we went over in class, to: remove time series that no longer exist, and add new time series.

## Transitions

You are also to ensure smooth animations when elements are added/removed via transitions. In particular, you will support the following:

* Time series transitions: if we are **expanding**, then the clicked time series should immediately be removed, while the added time series' positions should begin at the clicked time series, and then animate to their respective locations. If we are **collapsing**, then all time series that are to be removed should animate to the clicked time series, and the clicked time series should only appear when this animation has finished.
* Text transitions: have the opacity decrease to 0 if the time series for a text label is being removed, and increase to 1 if it is a text label for a new time series.

## Streamgraph (Graduate Students only)

Although adding interactivity to visualizing multiple time series is useful, occlusion remains a problem, particularly for the music dataset. **Streamgraphs** address this problem by stacking time series graphs and using area to encode time series values. Namely, the length, in the y-coordinate, of a time series identified by a given color gives us its **relative** value. Note that in streamgraphs we cannot display absolute values anymore, a trade-off we take in reducing clutter. You should create a new file for this assignment, titled `interactive_streamgraph.js`, that will build off of your solution to `interactive_time_series.js`.

There are several important choices to make regarding streamgraphs. Arguably most important is the **baseline** curve, the bottom curve that all other time series are stacked on top of. For this assignment you will use the approach of [ThemeRiver](http://www.ifs.tuwien.ac.at/~silvia/wien/vu-infovis/references/havre-ieeeinfovis00.pdf). The topic of baseline is eloquently discussed in the following paper by [Lee Byron and Martin Wattenberg](http://leebyron.com/streamgraph/stackedgraphs_byron_wattenberg.pdf). More specifically, you should implement the baseline outlined in Section 5.1 when discussing ThemeRiver, e.g. the sum of the time series, negated and scaled by 0.5. This will give a layout that is symmetric around the x-axis.

All other considerations regarding streamgraphs: time series ordering, layer labeling, color, you need not worry about. Simply color the areas by the colors you used in the previous part of the assignment. Furthermore, you do not need to address transitions - updates will be instantaneous. But the user should still be able to click on an area to expand, and Shift+click on on area to collapse. Last, text labels need not be animated either, but should be positioned in the center of each time series' area (in the y-axis).

Try out the two different aggregations: mean and sum. You should now see that it is much easier to perceive summations of time series compared to the above technique of visualizing individual time series, since in the latter we are not rescaling the axes.

## Hand-in

Zip up the `assignment2` directory with all html and javascript files, please exclude the data files when zipping, and submit it to Brightspace.

## Grading

* 10 points: hierarchical aggregation of time series
* 50 points: Visual Encoding
* 25 points: properly using data join: `enter` and `exit` selections for handling an arbitrary subset of nodes, and proper usage of keys
* 10 points: plot axes
* 10 points: text labels
* 5 points: color assigned to elements
* 30 points: Interactivity
* 10 points: determine nodes that to be expanded/collapsed
* 10 points: support expansion behavior in the data join
* 10 points: support collapse behavior in the data join
* 10 points: Transitions
* 8 points: path element transitions
* 2 points: text element transitions
* 30 points: Streamgraphs
* 10 points: computing a stacked representation of the count data
* 15 points: visualizing the stacked time series data in terms of area marks, colored by the time series' assigned colors.
* 5 points: text labels positioned properly

Please note for graduate students: your grade is out of 130 points, but still counts as the same amount as any other assignment in its contribution towards your overall grade.
10 changes: 10 additions & 0 deletions assignment2/basictheme.css
Original file line number Diff line number Diff line change
@@ -0,0 +1,10 @@
svg text {
cursor: default;
-webkit-user-select: none;
-moz-user-select: none;
-ms-user-select: none;
user-select: none;
}
svg text::selection {
background: none;
}
21 changes: 21 additions & 0 deletions assignment2/index.html
Original file line number Diff line number Diff line change
@@ -0,0 +1,21 @@
<!DOCTYPE html>
<html lang='en'>
<head>
<meta charset='utf-8'>
<title>Assignment 2 - Plotting Time Series Data</title>
<script type='text/javascript' src='../d3.js'></script>
<link rel="stylesheet" type="text/css" href="basictheme.css">
</head>
<body>
<script type='text/javascript' src='interactive_time_series.js'></script>
<script type='text/javascript'>
is_music_time_series = false;
json_filename = is_music_time_series ? 'music_time_series.json' : 'japan_population.json'
d3.json(json_filename)
.then(function(data) {
count_tree = data;
plot_it();
})
</script>
</body>
</html>
Loading

0 comments on commit 8cbb141

Please sign in to comment.