Home | History | Annotate | Download | only in tracing
      1 // Copyright (c) 2012 The Chromium Authors. All rights reserved.
      2 // Use of this source code is governed by a BSD-style license that can be
      3 // found in the LICENSE file.
      4 
      5 
      6 /**
      7  * @fileoverview Provides a mechanism for drawing massive numbers of
      8  * colored rectangles into a canvas in an efficient manner, provided
      9  * they are drawn left to right with fixed y and height throughout.
     10  *
     11  * The basic idea used here is to fuse subpixel rectangles together so that
     12  * we never issue a canvas fillRect for them. It turns out Javascript can
     13  * do this quite efficiently, compared to asking Canvas2D to do the same.
     14  *
     15  * A few extra things are done by this class in the name of speed:
     16  * - Viewport culling: off-viewport rectangles are discarded.
     17  *
     18  * - The actual discarding operation is done in world space,
     19  *   e.g. pre-transform.
     20  *
     21  * - Rather than expending compute cycles trying to figure out an average
     22  *   color for fused rectangles from css strings, you instead draw using
     23  *   palletized colors. The fused rect is the max pallete index encountered.
     24  *
     25  * Make sure to flush the trackRenderer before finishing drawing in order
     26  * to commit any queued drawing operations.
     27  */
     28 cr.define('tracing', function() {
     29 
     30   /**
     31    * Creates a fast rect renderer with a specific set of culling rules
     32    * and color pallette.
     33    * @param {GraphicsContext2D} ctx Canvas2D drawing context.
     34    * @param {number} vpLeft The leftmost visible part of the drawing viewport.
     35    * @param {number} minRectSize Only rectangles with width < minRectSize are
     36    *    considered for merging.
     37    * @param {number} maxMergeDist Controls how many successive small rectangles
     38    *    can be merged together before issuing a rectangle.
     39    * @param {number} vpRight The rightmost visible part of the viewport.
     40    * @param {Array} pallette The color pallete for drawing. Pallette slots
     41    *    should map to valid Canvas fillStyle strings.
     42    *
     43    * @constructor
     44    */
     45   function FastRectRenderer(ctx, vpLeft, minRectSize, maxMergeDist, vpRight,
     46                             pallette) {
     47     this.ctx_ = ctx;
     48     this.vpLeft_ = vpLeft;
     49     this.minRectSize_ = minRectSize;
     50     this.maxMergeDist_ = maxMergeDist;
     51     this.vpRight_ = vpRight;
     52     this.pallette_ = pallette;
     53   }
     54 
     55   FastRectRenderer.prototype = {
     56     y_: 0,
     57     h_: 0,
     58     merging_: false,
     59     mergeStartX_: 0,
     60     mergeCurRight_: 0,
     61 
     62     /**
     63      * Changes the y position and height for subsequent fillRect
     64      * calls. x and width are specifieid on the fillRect calls.
     65      */
     66     setYandH: function(y, h) {
     67       this.flush();
     68       this.y_ = y;
     69       this.h_ = h;
     70     },
     71 
     72     /**
     73      * Fills rectangle at the specified location, if visible. If the
     74      * rectangle is subpixel, it will be merged with adjacent rectangles.
     75      * The drawing operation may not take effect until flush is called.
     76      * @param {number} colorId The color of this rectangle, as an index
     77      *     in the renderer's color pallete.
     78      */
     79     fillRect: function(x, w, colorId) {
     80       var r = x + w;
     81       if (r < this.vpLeft_ || x > this.vpRight_) return;
     82       if (w < this.minRectSize_) {
     83         if (r - this.mergeStartX_ > this.maxMergeDist_)
     84           this.flush();
     85         if (!this.merging_) {
     86           this.merging_ = true;
     87           this.mergeStartX_ = x;
     88           this.mergeCurRight_ = r;
     89           this.mergedColorId = colorId;
     90         } else {
     91           this.mergeCurRight_ = r;
     92           this.mergedColorId = Math.max(this.mergedColorId, colorId);
     93         }
     94       } else {
     95         if (this.merging_)
     96           this.flush();
     97         this.ctx_.fillStyle = this.pallette_[colorId];
     98         this.ctx_.fillRect(x, this.y_, w, this.h_);
     99       }
    100     },
    101 
    102     /**
    103      * Commits any pending fillRect operations to the underlying graphics
    104      * context.
    105      */
    106     flush: function() {
    107       if (this.merging_) {
    108         this.ctx_.fillStyle = this.pallette_[this.mergedColorId];
    109         this.ctx_.fillRect(this.mergeStartX_, this.y_,
    110                            this.mergeCurRight_ - this.mergeStartX_, this.h_);
    111         this.merging_ = false;
    112       }
    113     }
    114   };
    115 
    116   return {
    117     FastRectRenderer: FastRectRenderer
    118   };
    119 
    120 });
    121