1359 lines
		
	
	
	
		
			34 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			1359 lines
		
	
	
	
		
			34 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| /**
 | |
|  * Bind polyfill from https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Function/bind#Compatibility
 | |
|  */
 | |
|   if (!Function.prototype.bind) {
 | |
|     Function.prototype.bind = function (oThis) {
 | |
|       if (typeof this !== "function") {
 | |
|         // closest thing possible to the ECMAScript 5 internal IsCallable function
 | |
|         throw new TypeError("Function.prototype.bind - what is trying to be bound is not callable");
 | |
|       }
 | |
| 
 | |
|       var aArgs = Array.prototype.slice.call(arguments, 1),
 | |
|       fToBind = this,
 | |
|       fNOP = function () {},
 | |
|       fBound = function () {
 | |
|         return fToBind.apply(this instanceof fNOP && oThis
 | |
|                              ? this
 | |
|                              : oThis,
 | |
|                              aArgs.concat(Array.prototype.slice.call(arguments)));
 | |
|       };
 | |
| 
 | |
|       fNOP.prototype = this.prototype;
 | |
|       fBound.prototype = new fNOP();
 | |
| 
 | |
|       return fBound;
 | |
|     };
 | |
|   }
 | |
| 
 | |
| var DEPTH_HEIGHT = 100;
 | |
| var DEPTH_WIDTH = 140;
 | |
| 
 | |
| /**
 | |
|  * Based off of http://mbostock.github.io/d3/talk/20111018/tree.html by Mike Bostock (@mbostock)
 | |
|  */
 | |
| function ImageHistoryTree(namespace, name, images, getTagsForImage, formatComment, formatTime,
 | |
|                           formatCommand, opt_tagFilter) {
 | |
| 
 | |
|   /**
 | |
|    * The namespace of the repo.
 | |
|    */
 | |
|   this.repoNamespace_ = namespace;
 | |
| 
 | |
|   /**
 | |
|    * The name of the repo.
 | |
|    */
 | |
|   this.repoName_ = name;
 | |
| 
 | |
|   /**
 | |
|    * The images to display.
 | |
|    */
 | |
|   this.images_ = images;
 | |
| 
 | |
|   /**
 | |
|    * Method to retrieve the tags for an image.
 | |
|    */
 | |
|   this.getTagsForImage_ = getTagsForImage;
 | |
| 
 | |
|   /**
 | |
|    * Method to invoke to format a comment for an image.
 | |
|    */
 | |
|   this.formatComment_ = formatComment;
 | |
| 
 | |
|   /**
 | |
|    * Method to invoke to format the time for an image.
 | |
|    */
 | |
|   this.formatTime_ = formatTime;
 | |
| 
 | |
|   /**
 | |
|    * Method to invoke to format the command for an image.
 | |
|    */
 | |
|   this.formatCommand_ = formatCommand;
 | |
| 
 | |
|   /**
 | |
|    * Method for filtering the tags and image paths displayed in the tree.
 | |
|    */
 | |
|   this.tagFilter_ = opt_tagFilter || function() { return true; };
 | |
| 
 | |
|   /**
 | |
|    * The current tag (if any).
 | |
|    */
 | |
|   this.currentTag_ = null;
 | |
| 
 | |
|   /**
 | |
|    * The current image (if any).
 | |
|    */
 | |
|   this.currentImage_ = null;
 | |
| 
 | |
|   /**
 | |
|    * The currently highlighted node (if any).
 | |
|    */
 | |
|   this.currentNode_ = null;
 | |
| 
 | |
|   /**
 | |
|    * Counter for creating unique IDs.
 | |
|    */
 | |
|   this.idCounter_ = 0;
 | |
| }
 | |
| 
 | |
| 
 | |
| /**
 | |
|  * Calculates the dimensions of the tree.
 | |
|  */
 | |
| ImageHistoryTree.prototype.calculateDimensions_ = function(container) {
 | |
|   var cw = document.getElementById(container).clientWidth;
 | |
|   var ch = this.maxHeight_ * (DEPTH_HEIGHT + 10);
 | |
| 
 | |
|   var margin = { top: 40, right: 20, bottom: 20, left: 80 };
 | |
|   var m = [margin.top, margin.right, margin.bottom, margin.left];
 | |
|   var w = cw - m[1] - m[3];
 | |
|   var h = ch - m[0] - m[2];
 | |
| 
 | |
|   return {
 | |
|     'w': w,
 | |
|     'h': h,
 | |
|     'm': m,
 | |
|     'cw': cw,
 | |
|     'ch': ch
 | |
|   };
 | |
| };
 | |
| 
 | |
| 
 | |
| ImageHistoryTree.prototype.setupOverscroll_ = function() {
 | |
|   var container = this.container_;
 | |
|   var that = this;
 | |
|   var overscroll = $('#' + container).overscroll();
 | |
| 
 | |
|   overscroll.on('overscroll:dragstart', function() {
 | |
|     $(that).trigger({
 | |
|       'type': 'hideTagMenu'
 | |
|     });
 | |
| 
 | |
|     $(that).trigger({
 | |
|       'type': 'hideImageMenu'
 | |
|     });
 | |
|   });
 | |
| 
 | |
|   overscroll.on('scroll', function() {
 | |
|     $(that).trigger({
 | |
|       'type': 'hideTagMenu'
 | |
|     });
 | |
| 
 | |
|     $(that).trigger({
 | |
|       'type': 'hideImageMenu'
 | |
|     });
 | |
|   });
 | |
| };
 | |
| 
 | |
| 
 | |
| /**
 | |
|  * Updates the dimensions of the tree.
 | |
|  */
 | |
| ImageHistoryTree.prototype.updateDimensions_ = function() {
 | |
|   var container = this.container_;
 | |
|   var dimensions = this.calculateDimensions_(container);
 | |
|   if (!dimensions) { return; }
 | |
| 
 | |
|   var m = dimensions.m;
 | |
|   var w = dimensions.w;
 | |
|   var h = dimensions.h;
 | |
|   var cw = dimensions.cw;
 | |
|   var ch = dimensions.ch;
 | |
| 
 | |
|   // Set the height of the container so that it never goes offscreen.
 | |
|   if (!$('#' + container).removeOverscroll) { return; }
 | |
| 
 | |
|   $('#' + container).removeOverscroll();
 | |
|   var viewportHeight = $(window).height();
 | |
|   var boundingBox = document.getElementById(container).getBoundingClientRect();
 | |
|   document.getElementById(container).style.maxHeight = (viewportHeight - boundingBox.top - 100) + 'px';
 | |
| 
 | |
|   this.setupOverscroll_();
 | |
| 
 | |
|   // Update the tree.
 | |
|   var rootSvg = this.rootSvg_;
 | |
|   var tree = this.tree_;
 | |
|   var vis = this.vis_;
 | |
| 
 | |
| 
 | |
|   var ow = w + m[1] + m[3];
 | |
|   var oh = h + m[0] + m[2];
 | |
|   rootSvg
 | |
|     .attr("width", ow)
 | |
|     .attr("height", oh)
 | |
|     .attr("style", "width: " + ow + "px; height: " + oh + "px");
 | |
| 
 | |
|   tree.size([w, h]);
 | |
|   vis.attr("transform", "translate(" + m[3] + "," + m[0] + ")");
 | |
| 
 | |
|   return dimensions;
 | |
| };
 | |
| 
 | |
| 
 | |
| /**
 | |
|  * Draws the tree.
 | |
|  */
 | |
| ImageHistoryTree.prototype.draw = function(container) {
 | |
|   // Build the root of the tree.
 | |
|   var result = this.buildRoot_();
 | |
|   this.maxWidth_ = result['maxWidth'];
 | |
|   this.maxHeight_ = result['maxHeight'];
 | |
| 
 | |
|   // Save the container.
 | |
|   this.container_ = container;
 | |
| 
 | |
|   if (!$('#' + container)[0]) {
 | |
|     this.container_ = null;
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   // Create the tree and all its components.
 | |
|   var tree = d3.layout.tree()
 | |
|     .separation(function() { return 2; });
 | |
| 
 | |
|   var diagonal = d3.svg.diagonal()
 | |
|     .projection(function(d) { return [d.x, d.y]; });
 | |
| 
 | |
|   var rootSvg = d3.select("#" + container).append("svg:svg")
 | |
|     .attr("class", "image-tree");
 | |
| 
 | |
|   var vis = rootSvg.append("svg:g");
 | |
|   var formatComment = this.formatComment_;
 | |
|   var formatTime = this.formatTime_;
 | |
|   var formatCommand = this.formatCommand_;
 | |
| 
 | |
|   var tip = d3.tip()
 | |
|     .attr('class', 'd3-tip')
 | |
|     .offset([-1, 24])
 | |
|     .direction('e')
 | |
|     .html(function(d) {
 | |
|       var html = '';
 | |
|       if (d.virtual) {
 | |
|         return d.name;
 | |
|       }
 | |
| 
 | |
|       if (d.collapsed) {
 | |
|         for (var i = 1; i < d.encountered.length; ++i) {
 | |
|           html += '<span>' + d.encountered[i].image.id.substr(0, 12) + '</span>';
 | |
|           html += '<span class="created">' + formatTime(d.encountered[i].image.created) + '</span>';
 | |
|         }
 | |
|         return html;
 | |
|       }
 | |
| 
 | |
|       if (!d.image) {
 | |
|         return '(This repository is empty)';
 | |
|       }
 | |
| 
 | |
|       if (d.image.comment) {
 | |
|         html += '<span class="comment">' + formatComment(d.image.comment) + '</span>';
 | |
|       }
 | |
|       if (d.image.command && d.image.command.length) {
 | |
|         html += '<span class="command info-line"><i class="fa fa-terminal"></i>' + formatCommand(d.image) + '</span>';
 | |
|       }
 | |
|       html += '<span class="created info-line"><i class="fa fa-calendar"></i>' + formatTime(d.image.created) + '</span>';
 | |
| 
 | |
|       var tags = d.tags || [];
 | |
|       html += '<span class="tooltip-tags tags">';
 | |
|       for (var i = 0; i < tags.length; ++i) {
 | |
|         var tag = tags[i];
 | |
|         var kind = 'default';
 | |
|         html += '<span class="label label-' + kind + ' tag" data-tag="' + tag + '">' + tag + '</span>';
 | |
|       }
 | |
|       html += '</span>';
 | |
| 
 | |
|       return html;
 | |
|     })
 | |
| 
 | |
|   vis.call(tip);
 | |
| 
 | |
|   // Save all the state created.
 | |
|   this.diagonal_ = diagonal;
 | |
|   this.vis_ = vis;
 | |
|   this.rootSvg_ = rootSvg;
 | |
|   this.tip_ = tip;
 | |
|   this.tree_ = tree;
 | |
| 
 | |
|   // Update the dimensions of the tree.
 | |
|   var dimensions = this.updateDimensions_();
 | |
|   if (!dimensions) {
 | |
|     return this;
 | |
|   }
 | |
| 
 | |
|   // Populate the tree.
 | |
|   this.root_.x0 = dimensions.cw / 2;
 | |
|   this.root_.y0 = 0;
 | |
| 
 | |
|   this.setTag_(this.currentTag_);
 | |
|   this.setupOverscroll_();
 | |
| 
 | |
|   return this;
 | |
| };
 | |
| 
 | |
| 
 | |
| /**
 | |
|  * Redraws the image history to fit the new size.
 | |
|  */
 | |
| ImageHistoryTree.prototype.notifyResized = function() {
 | |
|   this.updateDimensions_();
 | |
|   this.update_(this.root_);
 | |
| };
 | |
| 
 | |
| 
 | |
| /**
 | |
|  * Sets the current tag displayed in the tree.
 | |
|  */
 | |
| ImageHistoryTree.prototype.setTag = function(tagName) {
 | |
|   this.setTag_(tagName);
 | |
| };
 | |
| 
 | |
| 
 | |
| /**
 | |
|  * Sets the current image displayed in the tree.
 | |
|  */
 | |
| ImageHistoryTree.prototype.setImage = function(imageId) {
 | |
|   this.setImage_(imageId);
 | |
| };
 | |
| 
 | |
| 
 | |
| /**
 | |
|  * Updates the highlighted path in the tree.
 | |
|  */
 | |
| ImageHistoryTree.prototype.setHighlightedPath_ = function(image) {
 | |
|   if (this.currentNode_) {
 | |
|     this.markPath_(this.currentNode_, false);
 | |
|   }
 | |
| 
 | |
|   var imageByDockerId = this.imageByDockerId_;
 | |
|   var currentNode = imageByDockerId[image.id];
 | |
|   if (currentNode) {
 | |
|     this.markPath_(currentNode, true);
 | |
|     this.currentNode_ = currentNode;
 | |
|   }
 | |
| };
 | |
| 
 | |
| 
 | |
| /**
 | |
|  * Returns the ancestors of the given image.
 | |
|  */
 | |
| ImageHistoryTree.prototype.getAncestors_ = function(image) {
 | |
|   var ancestorsString = image.ancestors;
 | |
| 
 | |
|   // Remove the starting and ending /s.
 | |
|   ancestorsString = ancestorsString.substr(1, ancestorsString.length - 2);
 | |
| 
 | |
|   // Split based on /.
 | |
|   ancestors = ancestorsString.split('/');
 | |
|   return ancestors;
 | |
| };
 | |
| 
 | |
| 
 | |
| /**
 | |
|  * Sets the current tag displayed in the tree and raises the event that the tag
 | |
|  * was changed.
 | |
|  */
 | |
| ImageHistoryTree.prototype.changeTag_ = function(tagName) {
 | |
|   $(this).trigger({
 | |
|     'type': 'tagChanged',
 | |
|     'tag': tagName
 | |
|   });
 | |
|   this.setTag_(tagName);
 | |
| };
 | |
| 
 | |
| 
 | |
| /**
 | |
|  * Sets the current image displayed in the tree and raises the event that the image
 | |
|  * was changed.
 | |
|  */
 | |
| ImageHistoryTree.prototype.changeImage_ = function(imageId) {
 | |
|   $(this).trigger({
 | |
|     'type': 'imageChanged',
 | |
|     'image': this.findImage_(function(image) { return image.id == imageId; })
 | |
|   });
 | |
|   this.setImage_(imageId);
 | |
| };
 | |
| 
 | |
| 
 | |
| /**
 | |
|  * Expands the given collapsed node in the tree.
 | |
|  */
 | |
| ImageHistoryTree.prototype.expandCollapsed_ = function(imageNode) {
 | |
|   var index = imageNode.parent.children.indexOf(imageNode);
 | |
|   if (index < 0 || imageNode.encountered.length < 2) {
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   // Note: we start at 1 since the 0th encountered node is the parent.
 | |
|   imageNode.parent.children.splice(index, 1, imageNode.encountered[1]);
 | |
|   this.maxHeight_ = this.determineMaximumHeight_(this.root_);
 | |
|   this.update_(this.root_);
 | |
|   this.updateDimensions_();
 | |
| };
 | |
| 
 | |
| 
 | |
| /**
 | |
|  * Returns the level of the node in the tree. Recursively computes and updates
 | |
|  * if necessary.
 | |
|  */
 | |
| ImageHistoryTree.prototype.calculateLevel_ = function(node) {
 | |
|   if (node['level'] != null) {
 | |
|     return node['level'];
 | |
|   }
 | |
| 
 | |
|   if (node['parent'] == null) {
 | |
|     return node['level'] = 0;
 | |
|   }
 | |
| 
 | |
|   return node['level'] = (this.calculateLevel_(node['parent']) + 1);
 | |
| };
 | |
| 
 | |
| 
 | |
| /**
 | |
|  * Builds the root node for the tree.
 | |
|  */
 | |
| ImageHistoryTree.prototype.buildRoot_ = function() {
 | |
|   // Build the formatted JSON block for the tree. It must be of the form:
 | |
|   //  {
 | |
|   //    "name": "...",
 | |
|   //    "children": [...]
 | |
|   //  }
 | |
|   var formatted = {"name": "No images found"};
 | |
| 
 | |
|   // Build a node for each image.
 | |
|   var imageByDockerId = {};
 | |
|   for (var i = 0; i < this.images_.length; ++i) {
 | |
|     var image = this.images_[i];
 | |
| 
 | |
|     // Skip images that are currently uploading.
 | |
|     if (image.uploading) { continue; }
 | |
| 
 | |
|     var imageNode = {
 | |
|       "name": image.id.substr(0, 12),
 | |
|       "children": [],
 | |
|       "image": image,
 | |
|       "tags": this.getTagsForImage_(image),
 | |
|       "level": null
 | |
|     };
 | |
|     imageByDockerId[image.id] = imageNode;
 | |
|   }
 | |
|   this.imageByDockerId_ = imageByDockerId;
 | |
| 
 | |
|   // For each node, attach it to its immediate parent. If there is no immediate parent,
 | |
|   // then the node is the root.
 | |
|   var roots = [];
 | |
|   var nodeCountsByLevel = {};
 | |
|   for (var i = 0; i < this.images_.length; ++i) {
 | |
|     var image = this.images_[i];
 | |
| 
 | |
|     // Skip images that are currently uploading.
 | |
|     if (image.uploading) { continue; }
 | |
| 
 | |
|     var imageNode = imageByDockerId[image.id];
 | |
|     var ancestors = this.getAncestors_(image);
 | |
|     var immediateParent = ancestors[ancestors.length - 1];
 | |
|     var parent = imageByDockerId[immediateParent];
 | |
|     if (parent) {
 | |
|       // Add a reference to the parent. This makes walking the tree later easier.
 | |
|       imageNode.parent = parent;
 | |
|       parent.children.push(imageNode);
 | |
|     } else {
 | |
|       imageNode['level'] = 0;
 | |
|       roots.push(imageNode);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   // Calculate each node's level.
 | |
|   for (var i = 0; i < this.images_.length; ++i) {
 | |
|     var image = this.images_[i];
 | |
| 
 | |
|     // Skip images that are currently uploading.
 | |
|     if (image.uploading) { continue; }
 | |
| 
 | |
|     var imageNode = imageByDockerId[image.id];
 | |
|     var level = this.calculateLevel_(imageNode);
 | |
|     if (nodeCountsByLevel[level] == null) {
 | |
|       nodeCountsByLevel[level] = 1;
 | |
|     } else {
 | |
|       nodeCountsByLevel[level]++;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   // If there are multiple root nodes, then there is at least one branch without shared
 | |
|   // ancestry and we use the virtual node. Otherwise, we use the root node found.
 | |
|   var root = {
 | |
|     'name': '',
 | |
|     'children': roots,
 | |
|     'virtual': true
 | |
|   };
 | |
| 
 | |
|   if (roots.length == 1) {
 | |
|     root = roots[0];
 | |
|   }
 | |
| 
 | |
|   // Determine the maximum number of nodes at a particular level. This is used to size
 | |
|   // the width of the tree properly.
 | |
|   var maxChildCount = 0;
 | |
|   var maxChildHeight = 0;
 | |
|   Object.keys(nodeCountsByLevel).forEach(function(key){
 | |
|     maxChildCount = Math.max(maxChildCount, nodeCountsByLevel[key]);
 | |
|     maxChildHeight = Math.max(maxChildHeight, key);
 | |
|   });
 | |
| 
 | |
|   // Recursively prune the nodes that are not referenced by a tag
 | |
|   this.pruneUnreferenced_(root);
 | |
| 
 | |
|   // Compact the graph so that any single chain of three (or more) images becomes a collapsed
 | |
|   // section. We only do this if the max width is > 1 (since for a single width tree, no long
 | |
|   // chain will hide a branch).
 | |
|   if (maxChildCount > 1) {
 | |
|     this.collapseNodes_(root);
 | |
|   }
 | |
| 
 | |
|   // Determine the maximum height of the tree, with collapsed nodes.
 | |
|   var maxCollapsedHeight = this.determineMaximumHeight_(root);
 | |
| 
 | |
|   // Finally, set the root node and return.
 | |
|   this.root_ = root;
 | |
| 
 | |
|   return {
 | |
|     'maxWidth': maxChildCount + 1,
 | |
|     'maxHeight': maxCollapsedHeight
 | |
|   };
 | |
| };
 | |
| 
 | |
| 
 | |
| /**
 | |
|  * Prunes images which are not referenced either directly or indirectly by any tag.
 | |
|  */
 | |
| ImageHistoryTree.prototype.pruneUnreferenced_ = function(node) {
 | |
|   if (node.children) {
 | |
|     var surviving_children = []
 | |
|     for (var i = 0; i < node.children.length; ++i) {
 | |
|       if (!this.pruneUnreferenced_(node.children[i])) {
 | |
|         surviving_children.push(node.children[i]);
 | |
|       }
 | |
|     }
 | |
|     node.children = surviving_children;
 | |
|   }
 | |
| 
 | |
|   if (!node.tags) {
 | |
|     return node.children.length == 0;
 | |
|   }
 | |
| 
 | |
|   var tags = [];
 | |
|   for (var i = 0; i < node.tags.length; ++i) {
 | |
|     if (this.tagFilter_(node.tags[i])) {
 | |
|       tags.push(node.tags[i]);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   return (node.children.length == 0 && tags.length == 0);
 | |
| };
 | |
| 
 | |
| 
 | |
| /**
 | |
|  * Determines the height of the tree at its longest chain.
 | |
|  */
 | |
| ImageHistoryTree.prototype.determineMaximumHeight_ = function(node) {
 | |
|   var maxHeight = 0;
 | |
|   if (node.children) {
 | |
|     for (var i = 0; i < node.children.length; ++i) {
 | |
|       maxHeight = Math.max(this.determineMaximumHeight_(node.children[i]), maxHeight);
 | |
|     }
 | |
|   }
 | |
|   return maxHeight + 1;
 | |
| };
 | |
| 
 | |
| 
 | |
| /**
 | |
|  * Collapses long single chains of nodes (3 or more) into single nodes to make the graph more
 | |
|  * compact.
 | |
|  */
 | |
| ImageHistoryTree.prototype.collapseNodes_ = function(node) {
 | |
|   if (node.children && node.children.length == 1) {
 | |
|     // Keep searching downward until we find a node with more than a single child.
 | |
|     var current = node;
 | |
|     var previous = node;
 | |
|     var encountered = [];
 | |
|     while (current.children
 | |
|           && current.children.length == 1
 | |
|            && current.tags
 | |
|            && current.tags.length == 0) {
 | |
|       encountered.push(current);
 | |
|       previous = current;
 | |
|       current = current.children[0];
 | |
|     }
 | |
| 
 | |
|     if (encountered.length >= 3) {
 | |
|       // Collapse the node.
 | |
|       var collapsed = {
 | |
|         "name": '(' + (encountered.length - 1) + ' images)',
 | |
|         "children": [current],
 | |
|         "collapsed": true,
 | |
|         "encountered": encountered
 | |
|       };
 | |
|       node.children = [collapsed];
 | |
| 
 | |
|       // Update the parent relationships.
 | |
|       collapsed.parent = node;
 | |
|       current.parent = collapsed;
 | |
|       return;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (node.children) {
 | |
|     for (var i = 0; i < node.children.length; ++i) {
 | |
|       this.collapseNodes_(node.children[i]);
 | |
|     }
 | |
|   }
 | |
| };
 | |
| 
 | |
| 
 | |
| /**
 | |
|  * Determines the maximum child count for the node and its children.
 | |
|  */
 | |
| ImageHistoryTree.prototype.determineMaximumChildCount_ = function(node) {
 | |
|   var children = node.children;
 | |
|   var myLevelCount = children.length;
 | |
|   var nestedCount = 0;
 | |
| 
 | |
|   for (var i = 0; i < children.length; ++i) {
 | |
|     nestedCount += children[i].children.length;
 | |
|   }
 | |
| 
 | |
|   return Math.max(myLevelCount, nestedCount);
 | |
| };
 | |
| 
 | |
| 
 | |
| /**
 | |
|  * Finds the image where the checker function returns true and returns it or null
 | |
|  * if none.
 | |
|  */
 | |
| ImageHistoryTree.prototype.findImage_ = function(checker) {
 | |
|   for (var i = 0; i < this.images_.length; ++i) {
 | |
|     var image = this.images_[i];
 | |
|     if (checker(image)) {
 | |
|       return image;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   return null;
 | |
| };
 | |
| 
 | |
| 
 | |
| /**
 | |
|  * Marks the full node path from the given starting node on whether it is highlighted.
 | |
|  */
 | |
| ImageHistoryTree.prototype.markPath_ = function(startingNode, isHighlighted) {
 | |
|   var currentNode = startingNode;
 | |
|   currentNode.current = isHighlighted;
 | |
|   while (currentNode != null) {
 | |
|     currentNode.highlighted = isHighlighted;
 | |
|     currentNode = currentNode.parent;
 | |
|   }
 | |
| };
 | |
| 
 | |
| 
 | |
| /**
 | |
|  * Sets the current tag displayed in the tree.
 | |
|  */
 | |
| ImageHistoryTree.prototype.setTag_ = function(tagName) {
 | |
|   if (tagName == this.currentTag_) {
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   var imageByDockerId = this.imageByDockerId_;
 | |
| 
 | |
|   // Save the current tag.
 | |
|   var previousTagName = this.currentTag_;
 | |
|   this.currentTag_ = tagName;
 | |
|   this.currentImage_ = null;
 | |
| 
 | |
|   // Update the path.
 | |
|   var that = this;
 | |
|   var tagImage = this.findImage_(function(image) {
 | |
|     return tagName && (that.getTagsForImage_(image).indexOf(tagName) >= 0);
 | |
|   });
 | |
| 
 | |
|   if (tagImage) {
 | |
|     this.setHighlightedPath_(tagImage);
 | |
|   }
 | |
| 
 | |
|   // Ensure that the children are in the correct order.
 | |
|   for (var i = 0; i < this.images_.length; ++i) {
 | |
|     var image = this.images_[i];
 | |
| 
 | |
|     // Skip images that are currently uploading.
 | |
|     if (image.uploading) { continue; }
 | |
| 
 | |
|     var imageNode = this.imageByDockerId_[image.id];
 | |
|     var ancestors = this.getAncestors_(image);
 | |
|     var immediateParent = ancestors[ancestors.length - 1];
 | |
|     var parent = imageByDockerId[immediateParent];
 | |
|     if (parent && imageNode.highlighted) {
 | |
|       var arr = parent.children;
 | |
|       if (parent._children) {
 | |
|         arr = parent._children;
 | |
|       }
 | |
| 
 | |
|       if (arr[0] != imageNode) {
 | |
|         var index = arr.indexOf(imageNode);
 | |
|         if (index > 0) {
 | |
|           arr.splice(index, 1);
 | |
|           arr.splice(0, 0, imageNode);
 | |
|         }
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   // Update the tree.
 | |
|   this.update_(this.root_);
 | |
| };
 | |
| 
 | |
| 
 | |
| /**
 | |
|  * Sets the current image highlighted in the tree.
 | |
|  */
 | |
| ImageHistoryTree.prototype.setImage_ = function(imageId) {
 | |
|   // Find the new current image.
 | |
|   var newImage = this.findImage_(function(image) {
 | |
|     return image.id == imageId;
 | |
|   });
 | |
| 
 | |
|   if (newImage == this.currentImage_) {
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   this.setHighlightedPath_(newImage);
 | |
|   this.currentImage_ = newImage;
 | |
|   this.currentTag_ = null;
 | |
|   this.update_(this.root_);
 | |
| };
 | |
| 
 | |
| 
 | |
| /**
 | |
|  * Updates the tree in response to a click on a node.
 | |
|  */
 | |
| ImageHistoryTree.prototype.update_ = function(source) {
 | |
|   var tree = this.tree_;
 | |
|   var vis = this.vis_;
 | |
|   var diagonal = this.diagonal_;
 | |
|   var tip = this.tip_;
 | |
|   var currentTag = this.currentTag_;
 | |
|   var currentImage = this.currentImage_;
 | |
|   var repoNamespace = this.repoNamespace_;
 | |
|   var repoName = this.repoName_;
 | |
|   var maxHeight = this.maxHeight_;
 | |
| 
 | |
|   var that = this;
 | |
| 
 | |
|   var duration = 500;
 | |
| 
 | |
|   // Compute the new tree layout.
 | |
|   var nodes = tree.nodes(this.root_).reverse();
 | |
| 
 | |
|   // Normalize for fixed-depth.
 | |
|   nodes.forEach(function(d) { d.y = (maxHeight - d.depth - 1) * DEPTH_HEIGHT; });
 | |
| 
 | |
|   // Update the nodes...
 | |
|   var node = vis.selectAll("g.node")
 | |
|     .data(nodes, function(d) { return d.id || (d.id = that.idCounter_++); });
 | |
| 
 | |
|   // Enter any new nodes at the parent's previous position.
 | |
|   var nodeEnter = node.enter().append("svg:g")
 | |
|     .attr("class", "node")
 | |
|     .attr("transform", function(d) { return "translate(" + source.x0 + "," + source.y0 + ")"; });
 | |
| 
 | |
|   nodeEnter.append("svg:circle")
 | |
|     .attr("r", 1e-6)
 | |
|     .style("fill", function(d) { return d._children ? "lightsteelblue" : "#fff"; })
 | |
|     .on("click", function(d) { that.toggle_(d); that.update_(d); });
 | |
| 
 | |
|   // Create the group that will contain the node name and its tags.
 | |
|   var g = nodeEnter.append("svg:g").style("fill-opacity", 1e-6);
 | |
| 
 | |
|   // Add the repo ID.
 | |
|   g.append("svg:text")
 | |
|     .attr("x", function(d) { return d.children || d._children ? -10 : 10; })
 | |
|     .attr("dy", ".35em")
 | |
|     .attr("text-anchor", function(d) { return d.children || d._children ? "end" : "start"; })
 | |
|     .text(function(d) { return d.name; })
 | |
|     .on("click", function(d) {
 | |
|       if (d.image) { that.changeImage_(d.image.id); }
 | |
|       if (d.collapsed) { that.expandCollapsed_(d); }
 | |
|     })
 | |
|     .on('mouseover', tip.show)
 | |
|     .on('mouseout', tip.hide)
 | |
|     .on("contextmenu", function(d, e) {
 | |
|       d3.event.preventDefault();
 | |
| 
 | |
|       if (d.image) {
 | |
|         $(that).trigger({
 | |
|           'type': 'showImageMenu',
 | |
|           'image': d.image.id,
 | |
|           'clientX': d3.event.clientX,
 | |
|           'clientY': d3.event.clientY
 | |
|         });
 | |
|       }
 | |
|     });
 | |
| 
 | |
|   nodeEnter.selectAll("tags")
 | |
|     .append("svg:text")
 | |
|     .text("bar");
 | |
| 
 | |
|   // Create the foreign object to hold the tags (if any).
 | |
|   var fo = g.append("svg:foreignObject")
 | |
|     .attr("class", "fo")
 | |
|     .attr("x", 14)
 | |
|     .attr("y", 12)
 | |
|     .attr("width", 110)
 | |
|     .attr("height", DEPTH_HEIGHT - 20);
 | |
| 
 | |
|   // Add the tags container.
 | |
|   fo.append('xhtml:div')
 | |
|     .attr("class", "tags")
 | |
|     .style("display", "none");
 | |
| 
 | |
|   // Translate the foreign object so the tags are under the ID.
 | |
|   fo.attr("transform", function(d, i) {
 | |
|     return "translate(" + [-130, 0 ] + ")";
 | |
|   });
 | |
| 
 | |
|   // Transition nodes to their new position.
 | |
|   var nodeUpdate = node.transition()
 | |
|     .duration(duration)
 | |
|     .attr("transform", function(d) { return "translate(" + d.x + "," + d.y + ")"; });
 | |
| 
 | |
|   // Update the node circle.
 | |
|   nodeUpdate.select("circle")
 | |
|     .attr("r", 4.5)
 | |
|     .attr("class", function(d) {
 | |
|       return (d._children ? "closed " : "open ") + (d.current ? "current " : "") + (d.highlighted ? "highlighted " : "");
 | |
|     })
 | |
|     .style("fill", function(d) {
 | |
|       if (d.current) {
 | |
|         return "";
 | |
|       }
 | |
|       return d._children ? "lightsteelblue" : "#fff";
 | |
|     });
 | |
| 
 | |
|   // Update the repo text.
 | |
|   nodeUpdate.select("text")
 | |
|     .attr("class", function(d) {
 | |
|       if (d.collapsed) {
 | |
|         return 'collapsed';
 | |
|       }
 | |
|       if (d.virtual) {
 | |
|         return 'virtual';
 | |
|       }
 | |
|       if (!currentImage) {
 | |
| 	return '';
 | |
|       }
 | |
|       return d.image.id == currentImage.id ? 'current' : '';
 | |
|     });
 | |
| 
 | |
|   // Ensure that the node is visible.
 | |
|   nodeUpdate.select("g")
 | |
|     .style("fill-opacity", 1);
 | |
| 
 | |
|   // Update the tags.
 | |
|   node.select(".tags")
 | |
|     .html(function(d) {
 | |
|       if (!d.tags) {
 | |
|         return '';
 | |
|       }
 | |
| 
 | |
|       var html = '<div style="width: ' + DEPTH_HEIGHT + 'px">';
 | |
|       for (var i = 0; i < d.tags.length; ++i) {
 | |
|         var tag = d.tags[i];
 | |
|         var kind = 'default';
 | |
|         if (tag == currentTag) {
 | |
|           kind = 'success';
 | |
|         }
 | |
|         html += '<span class="label label-' + kind + ' tag" data-tag="' + tag + '" title="' + tag + '" style="max-width: ' + DEPTH_HEIGHT + 'px">' + tag + '</span>';
 | |
|       }
 | |
|       html += '</div>';
 | |
|       return html;
 | |
|     });
 | |
| 
 | |
|   // Listen for click events on the labels.
 | |
|   node.selectAll(".tag")
 | |
|     .on("click", function(d, e) {
 | |
|       var tag = this.getAttribute('data-tag');
 | |
|       if (tag) {
 | |
|         that.changeTag_(tag);
 | |
|       }
 | |
|     })
 | |
|     .on("contextmenu", function(d, e) {
 | |
|       d3.event.preventDefault();
 | |
| 
 | |
|       var tag = this.getAttribute('data-tag');
 | |
|       if (tag) {
 | |
|         $(that).trigger({
 | |
|           'type': 'showTagMenu',
 | |
|           'tag': tag,
 | |
|           'clientX': d3.event.clientX,
 | |
|           'clientY': d3.event.clientY
 | |
|         });
 | |
|       }
 | |
|     });
 | |
| 
 | |
|   // Ensure the tags are visible.
 | |
|   nodeUpdate.select(".tags")
 | |
|     .style("display", "")
 | |
| 
 | |
|   // There is a bug in Chrome which sometimes prevents the foreignObject from redrawing. To that end,
 | |
|   // we force a redraw by adjusting the height of the object ever so slightly.
 | |
|   nodeUpdate.select(".fo")
 | |
|     .attr('height', function(d) {
 | |
|       return DEPTH_HEIGHT - 20 + Math.random() / 10;
 | |
|     });
 | |
| 
 | |
|   // Transition exiting nodes to the parent's new position.
 | |
|   var nodeExit = node.exit().transition()
 | |
|     .duration(duration)
 | |
|     .attr("transform", function(d) { return "translate(" + source.x + "," + source.y + ")"; })
 | |
|     .remove();
 | |
| 
 | |
|   nodeExit.select("circle")
 | |
|     .attr("r", 1e-6);
 | |
| 
 | |
|   nodeExit.select(".tags")
 | |
|     .style("display", "none");
 | |
| 
 | |
|   nodeExit.select("g")
 | |
|     .style("fill-opacity", 1e-6);
 | |
| 
 | |
|   // Update the links...
 | |
|   var link = vis.selectAll("path.link")
 | |
|     .data(tree.links(nodes), function(d) { return d.target.id; });
 | |
| 
 | |
|   // Enter any new links at the parent's previous position.
 | |
|   link.enter().insert("svg:path", "g")
 | |
|     .attr("class", function(d) {
 | |
|       var isHighlighted = d.target.highlighted;
 | |
|       return "link " + (isHighlighted ? "highlighted": "");
 | |
|     })
 | |
|     .attr("d", function(d) {
 | |
|       var o = {x: source.x0, y: source.y0};
 | |
|       return diagonal({source: o, target: o});
 | |
|     })
 | |
|     .transition()
 | |
|     .duration(duration)
 | |
|     .attr("d", diagonal);
 | |
| 
 | |
|   // Transition links to their new position.
 | |
|   link.transition()
 | |
|     .duration(duration)
 | |
|     .attr("d", diagonal)
 | |
|     .attr("class", function(d) {
 | |
|       var isHighlighted = d.target.highlighted;
 | |
|       return "link " + (isHighlighted ? "highlighted": "");
 | |
|     });
 | |
| 
 | |
|   // Transition exiting nodes to the parent's new position.
 | |
|   link.exit().transition()
 | |
|     .duration(duration)
 | |
|     .attr("d", function(d) {
 | |
|       var o = {x: source.x, y: source.y};
 | |
|       return diagonal({source: o, target: o});
 | |
|     })
 | |
|     .remove();
 | |
| 
 | |
|   // Stash the old positions for transition.
 | |
|   nodes.forEach(function(d) {
 | |
|     d.x0 = d.x;
 | |
|     d.y0 = d.y;
 | |
|   });
 | |
| };
 | |
| 
 | |
| 
 | |
| /**
 | |
|  * Toggles children of a node.
 | |
|  */
 | |
| ImageHistoryTree.prototype.toggle_ = function(d) {
 | |
|   if (d.children) {
 | |
|     d._children = d.children;
 | |
|     d.children = null;
 | |
|   } else {
 | |
|     d.children = d._children;
 | |
|     d._children = null;
 | |
|   }
 | |
| };
 | |
| 
 | |
| 
 | |
| /**
 | |
|  * Disposes of the tree.
 | |
|  */
 | |
| ImageHistoryTree.prototype.dispose = function() {
 | |
|   var container =  this.container_ ;
 | |
|   $('#' + container).removeOverscroll();
 | |
|   $('#' + container).html('');
 | |
| };
 | |
| 
 | |
| ////////////////////////////////////////////////////////////////////////////////
 | |
| 
 | |
| /**
 | |
|  * Based off of http://bl.ocks.org/mbostock/1346410
 | |
|  */
 | |
| function UsageChart() {
 | |
|   this.total_ = null;
 | |
|   this.count_ = null;
 | |
|   this.drawn_ = false;
 | |
| }
 | |
| 
 | |
| 
 | |
| /**
 | |
|  * Updates the chart with the given count and total of number of repositories.
 | |
|  */
 | |
| UsageChart.prototype.update = function(count, total) {
 | |
|   if (!this.g_) { return; }
 | |
|   this.total_ = total;
 | |
|   this.count_ = count;
 | |
|   this.drawInternal_();
 | |
| };
 | |
| 
 | |
| 
 | |
| /**
 | |
|  * Conducts the actual draw or update (if applicable).
 | |
|  */
 | |
| UsageChart.prototype.drawInternal_ = function() {
 | |
|   // If the total is null, then we have not yet set the proper counts.
 | |
|   if (this.total_ === null) { return; }
 | |
| 
 | |
|   var duration = 750;
 | |
| 
 | |
|   var arc = this.arc_;
 | |
|   var pie = this.pie_;
 | |
|   var arcTween = this.arcTween_;
 | |
| 
 | |
|   var color = d3.scale.category20();
 | |
|   var count = this.count_;
 | |
|   var total = this.total_;
 | |
| 
 | |
|   var data = [Math.max(count, 1), Math.max(0, total - count)];
 | |
| 
 | |
|   var arcTween = function(a) {
 | |
|     var i = d3.interpolate(this._current, a);
 | |
|     this._current = i(0);
 | |
|     return function(t) {
 | |
|       return arc(i(t));
 | |
|     };
 | |
|   };
 | |
| 
 | |
|   if (!this.drawn_) {
 | |
|     var text = this.g_.append("svg:text")
 | |
|       .attr("dy", 10)
 | |
|       .attr("dx", 0)
 | |
|       .attr('dominant-baseline', 'auto')
 | |
|       .attr('text-anchor', 'middle')
 | |
|       .attr('class', 'count-text')
 | |
|       .text(this.count_ + ' / ' + this.total_);
 | |
| 
 | |
|     var path = this.g_.datum(data).selectAll("path")
 | |
|       .data(pie)
 | |
|       .enter().append("path")
 | |
|       .attr("fill", function(d, i) { return color(i); })
 | |
|       .attr("class", function(d, i) { return 'arc-' + i; })
 | |
|       .attr("d", arc)
 | |
|       .each(function(d) { this._current = d; }); // store the initial angles
 | |
| 
 | |
|     this.path_ = path;
 | |
|     this.text_ = text;
 | |
|   } else {
 | |
|     pie.value(function(d, i) { return data[i]; }); // change the value function
 | |
|     this.path_ = this.path_.data(pie); // compute the new angles
 | |
|     this.path_.transition().duration(duration).attrTween("d", arcTween); // redraw the arcs
 | |
| 
 | |
|     // Update the text.
 | |
|     this.text_.text(this.count_ + ' / ' + this.total_);
 | |
|   }
 | |
| 
 | |
|   this.drawn_ = true;
 | |
| };
 | |
| 
 | |
| 
 | |
| /**
 | |
|  * Draws the chart in the given container.
 | |
|  */
 | |
| UsageChart.prototype.draw = function(container) {
 | |
|   var cw = 200;
 | |
|   var ch = 200;
 | |
|   var radius = Math.min(cw, ch) / 2;
 | |
| 
 | |
|   var pie = d3.layout.pie().sort(null);
 | |
| 
 | |
|   var arc = d3.svg.arc()
 | |
|     .innerRadius(radius - 50)
 | |
|     .outerRadius(radius - 25);
 | |
| 
 | |
|   var svg = d3.select("#" + container).append("svg:svg")
 | |
|     .attr("width", cw)
 | |
|     .attr("height", ch);
 | |
| 
 | |
|   var g = svg.append("g")
 | |
|     .attr("transform", "translate(" + cw / 2 + "," + ch / 2 + ")");
 | |
| 
 | |
|   this.svg_ = svg;
 | |
|   this.g_ = g;
 | |
|   this.pie_ = pie;
 | |
|   this.arc_ = arc;
 | |
|   this.width_ = cw;
 | |
|   this.drawInternal_();
 | |
| };
 | |
| 
 | |
| 
 | |
| ////////////////////////////////////////////////////////////////////////////////
 | |
| 
 | |
| /**
 | |
|  * A chart which displays the last seven days of actions in the account.
 | |
|  */
 | |
| function LogUsageChart(titleMap) {
 | |
|   this.titleMap_ = titleMap;
 | |
|   this.colorScale_ = d3.scale.category20();
 | |
|   this.entryMap_ = {};
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Builds the D3-representation of the data.
 | |
|  */
 | |
| LogUsageChart.prototype.buildData_ = function(aggregatedLogs) {
 | |
|   var parseDate = d3.time.format("%a, %d %b %Y %H:%M:%S %Z").parse
 | |
| 
 | |
|   // Build entries for each kind of event that occurred, on each day. We have one
 | |
|   // entry per {kind, day} pair.
 | |
|   var entries = [];
 | |
|   for (var i = 0; i < aggregatedLogs.length; ++i) {
 | |
|     var aggregated = aggregatedLogs[i];
 | |
|     var title = this.titleMap_[aggregated.kind] || aggregated.kind;
 | |
|     var datetime = parseDate(aggregated.datetime);
 | |
|     var dateDay = datetime.getDate();
 | |
|     if (dateDay < 10) {
 | |
|       dateDay = '0' + dateDay;
 | |
|     }
 | |
| 
 | |
|     var formatted = (datetime.getMonth() + 1) + '/' + dateDay;
 | |
|     var adjusted = new Date(datetime.getFullYear(), datetime.getMonth(), datetime.getDate());
 | |
|     var key = title + '_' + formatted;
 | |
|     var entry = {
 | |
|       'kind': aggregated.kind,
 | |
|       'title': title,
 | |
|       'adjusted': adjusted,
 | |
|       'formatted': datetime.getDate(),
 | |
|       'count': aggregated.count
 | |
|     };
 | |
| 
 | |
|     entries.push(entry);
 | |
|     this.entryMap_[key] = entry;
 | |
|   }
 | |
| 
 | |
|   // Build the data itself. We create a single entry for each possible kind of data, and then add (x, y) pairs
 | |
|   // for the number of times that kind of event occurred on a particular day.
 | |
|   var dataArray = [];
 | |
|   var dataMap = {};
 | |
|   var dateMap = {};
 | |
| 
 | |
|   for (var i = 0; i < entries.length; ++i) {
 | |
|     var entry = entries[i];
 | |
|     var key = entry.title;
 | |
|     var found = dataMap[key];
 | |
|     if (!found) {
 | |
|       found = {'key': key, 'values': [], 'kind': entry.kind};
 | |
|       dataMap[key] = found;
 | |
|       dataArray.push(found);
 | |
|     }
 | |
| 
 | |
|     found.values.push({
 | |
|       'x': entry.adjusted,
 | |
|       'y': entry.count
 | |
|     });
 | |
| 
 | |
|     dateMap[entry.adjusted.toString()] = entry.adjusted;
 | |
|   }
 | |
| 
 | |
|   // Note: nvd3 has a bug that causes d3 to fail if there is not an entry for every single
 | |
|   // kind on each day that has data. Therefore, we pad those days with 0-length entries for each
 | |
|   // kind.
 | |
|   for (var i = 0; i < dataArray.length; ++i) {
 | |
|     var datum = dataArray[i];
 | |
|     for (var sDate in dateMap) {
 | |
|       if (!dateMap.hasOwnProperty(sDate)) {
 | |
|         continue;
 | |
|       }
 | |
| 
 | |
|       var cDate = dateMap[sDate];
 | |
|       var found = false;
 | |
|       for (var j = 0; j < datum.values.length; ++j) {
 | |
|         if (datum.values[j]['x'].getDate() == cDate.getDate()) {
 | |
|           found = true;
 | |
|           break;
 | |
|         }
 | |
|       }
 | |
| 
 | |
|       if (!found) {
 | |
|         datum.values.push({
 | |
|           'x': cDate,
 | |
|           'y': 0
 | |
|         });
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     datum.values.sort(function(a, b) {
 | |
|       return a['x'].getDate() - b['x'].getDate();
 | |
|     });
 | |
|   }
 | |
| 
 | |
|   return this.data_ = dataArray;
 | |
| };
 | |
| 
 | |
| 
 | |
| /**
 | |
|  * Renders the tooltip when hovering over an element in the chart.
 | |
|  */
 | |
| LogUsageChart.prototype.renderTooltip_ = function(d, e) {
 | |
|   if (e[0] == '0') {
 | |
|     e = e.substr(1);
 | |
|   }
 | |
| 
 | |
|   var key = d + '_' + e;
 | |
|   var entry = this.entryMap_[key];
 | |
|   if (!entry) {
 | |
|     entry = {'count': 0};
 | |
|   }
 | |
| 
 | |
|   var s = entry.count == 1 ? '' : 's';
 | |
|   return d + ' - ' + entry.count + ' time' + s + ' on ' + e;
 | |
| };
 | |
| 
 | |
| 
 | |
| /**
 | |
|  * Returns the color used in the chart for log entries of the given
 | |
|  * kind.
 | |
|  */
 | |
| LogUsageChart.prototype.getColor = function(kind) {
 | |
|   var colors = this.colorScale_.range();
 | |
|   var index = 0;
 | |
|   for (var i = 0; i < this.data_.length; ++i) {
 | |
|     var datum = this.data_[i];
 | |
|     var key = this.titleMap_[kind] || kind;
 | |
|     if (datum.key == key) {
 | |
|       index = i;
 | |
|       break;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   return colors[index];
 | |
| };
 | |
| 
 | |
| 
 | |
| /**
 | |
|  * Handler for when an element in the chart has been clicked.
 | |
|  */
 | |
| LogUsageChart.prototype.handleElementClicked_ = function(e) {
 | |
|   var key = e.series.key;
 | |
|   var kind = e.series.kind;
 | |
|   var disabled = [];
 | |
| 
 | |
|   var enabledCount = 0;
 | |
|   var d = this.chart_.multibar.disabled();
 | |
|   for (var i = 0; i < this.data_.length; ++i) {
 | |
|     enabledCount += (d[i] ? 0 : 1);
 | |
|   }
 | |
| 
 | |
|   for (var i = 0; i < this.data_.length; ++i) {
 | |
|     disabled.push(enabledCount == 1 ? false : this.data_[i].key != key);
 | |
|   }
 | |
| 
 | |
|   var allowed = {};
 | |
|   allowed[kind] = true;
 | |
| 
 | |
|   this.chart_.dispatch.changeState({ 'disabled': disabled });
 | |
|   $(this).trigger({
 | |
|     'type': 'filteringChanged',
 | |
|     'allowed': enabledCount == 1 ? null : allowed
 | |
|   });
 | |
| };
 | |
| 
 | |
| 
 | |
| /**
 | |
|  * Handler for when the state of the chart has changed.
 | |
|  */
 | |
| LogUsageChart.prototype.handleStateChange_ = function(e) {
 | |
|   var allowed = {};
 | |
|   var disabled = e.disabled;
 | |
|   for (var i = 0; i < this.data_.length; ++i) {
 | |
|     if (!disabled[i]) {
 | |
|       allowed[this.data_[i].kind] = true;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   $(this).trigger({
 | |
|     'type': 'filteringChanged',
 | |
|     'allowed': allowed
 | |
|   });
 | |
| };
 | |
| 
 | |
| 
 | |
| /**
 | |
|  * Draws the chart in the given container element.
 | |
|  */
 | |
| LogUsageChart.prototype.draw = function(container, logData, startDate, endDate) {
 | |
|   // Reset the container's contents.
 | |
|   var containerElm = document.getElementById(container);
 | |
|   if (!containerElm) {
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   containerElm.innerHTML = '<svg></svg>';
 | |
| 
 | |
|   // Returns a date offset from the given date by "days" Days.
 | |
|   var offsetDate = function(d, days) {
 | |
|     var copy = new Date(d.getTime());
 | |
|     copy.setDate(copy.getDate() + days);
 | |
|     return copy;
 | |
|   };
 | |
| 
 | |
|   var that = this;
 | |
|   var data = this.buildData_(logData);
 | |
|   nv.addGraph(function() {
 | |
|     // Build the chart itself.
 | |
|     var chart = nv.models.multiBarChart()
 | |
|       .margin({top: 30, right: 30, bottom: 50, left: 60})
 | |
|       .stacked(false)
 | |
|       .staggerLabels(false)
 | |
|       .tooltip(function(d, e) {
 | |
|         return that.renderTooltip_(d, e);
 | |
|       })
 | |
|       .color(that.colorScale_.range())
 | |
|       .groupSpacing(0.1);
 | |
| 
 | |
|     chart.multibar.delay(0);
 | |
| 
 | |
|     // Create the x-axis domain to encompass the full date range.
 | |
|     var domain = [];
 | |
|     var datetime = startDate;
 | |
|     while (datetime <= endDate) {
 | |
|       domain.push(datetime);
 | |
|       datetime = offsetDate(datetime, 1);
 | |
|     }
 | |
| 
 | |
|     chart.xDomain(domain);
 | |
| 
 | |
|     // Finish setting up the chart.
 | |
|     chart.xAxis
 | |
|       .tickFormat(d3.time.format("%m/%d"));
 | |
| 
 | |
|     chart.yAxis
 | |
|       .tickFormat(d3.format(',f'));
 | |
| 
 | |
|     d3.select('#bar-chart svg')
 | |
|       .datum(data)
 | |
|       .transition()
 | |
|       .duration(500)
 | |
|       .call(chart);
 | |
| 
 | |
|     nv.utils.windowResize(chart.update);
 | |
| 
 | |
|     chart.multibar.dispatch.on('elementClick', function(e) { that.handleElementClicked_(e); });
 | |
|     chart.dispatch.on('stateChange', function(e) { that.handleStateChange_(e); });
 | |
|     return that.chart_ = chart;
 | |
|   });
 | |
| };
 |