Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(97)

Side by Side Diff: third_party/WebKit/Source/devtools/front_end/ui_lazy/TimelineGrid.js

Issue 2623743002: DevTools: extract modules (non-extensions) (Closed)
Patch Set: rebaseline Created 3 years, 11 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch
OLDNEW
(Empty)
1 /*
2 * Copyright (C) 2007, 2008 Apple Inc. All rights reserved.
3 * Copyright (C) 2008, 2009 Anthony Ricaud <rik@webkit.org>
4 * Copyright (C) 2009 Google Inc. All rights reserved.
5 *
6 * Redistribution and use in source and binary forms, with or without
7 * modification, are permitted provided that the following conditions
8 * are met:
9 *
10 * 1. Redistributions of source code must retain the above copyright
11 * notice, this list of conditions and the following disclaimer.
12 * 2. Redistributions in binary form must reproduce the above copyright
13 * notice, this list of conditions and the following disclaimer in the
14 * documentation and/or other materials provided with the distribution.
15 * 3. Neither the name of Apple Computer, Inc. ("Apple") nor the names of
16 * its contributors may be used to endorse or promote products derived
17 * from this software without specific prior written permission.
18 *
19 * THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY
20 * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
21 * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
22 * DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY
23 * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
24 * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
25 * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
26 * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
27 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
28 * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
29 */
30
31 /**
32 * @unrestricted
33 */
34 UI.TimelineGrid = class {
35 constructor() {
36 this.element = createElement('div');
37 UI.appendStyle(this.element, 'ui_lazy/timelineGrid.css');
38
39 this._dividersElement = this.element.createChild('div', 'resources-dividers' );
40
41 this._gridHeaderElement = createElement('div');
42 this._gridHeaderElement.classList.add('timeline-grid-header');
43 this._eventDividersElement = this._gridHeaderElement.createChild('div', 'res ources-event-dividers');
44 this._dividersLabelBarElement = this._gridHeaderElement.createChild('div', ' resources-dividers-label-bar');
45 this.element.appendChild(this._gridHeaderElement);
46 }
47
48 /**
49 * @param {!UI.TimelineGrid.Calculator} calculator
50 * @param {number=} freeZoneAtLeft
51 * @return {!{offsets: !Array.<number>, precision: number}}
52 */
53 static calculateDividerOffsets(calculator, freeZoneAtLeft) {
54 /** @const */ var minGridSlicePx = 64; // minimal distance between grid lin es.
55
56 var clientWidth = calculator.computePosition(calculator.maximumBoundary());
57 var dividersCount = clientWidth / minGridSlicePx;
58 var gridSliceTime = calculator.boundarySpan() / dividersCount;
59 var pixelsPerTime = clientWidth / calculator.boundarySpan();
60
61 // Align gridSliceTime to a nearest round value.
62 // We allow spans that fit into the formula: span = (1|2|5)x10^n,
63 // e.g.: ... .1 .2 .5 1 2 5 10 20 50 ...
64 // After a span has been chosen make grid lines at multiples of the span.
65
66 var logGridSliceTime = Math.ceil(Math.log(gridSliceTime) / Math.LN10);
67 gridSliceTime = Math.pow(10, logGridSliceTime);
68 if (gridSliceTime * pixelsPerTime >= 5 * minGridSlicePx)
69 gridSliceTime = gridSliceTime / 5;
70 if (gridSliceTime * pixelsPerTime >= 2 * minGridSlicePx)
71 gridSliceTime = gridSliceTime / 2;
72
73 var leftBoundaryTime = calculator.minimumBoundary() - calculator.paddingLeft () / pixelsPerTime;
74 var firstDividerTime =
75 Math.ceil((leftBoundaryTime - calculator.zeroTime()) / gridSliceTime) * gridSliceTime + calculator.zeroTime();
76 var lastDividerTime = calculator.maximumBoundary();
77 // Add some extra space past the right boundary as the rightmost divider lab el text
78 // may be partially shown rather than just pop up when a new rightmost divid er gets into the view.
79 lastDividerTime += minGridSlicePx / pixelsPerTime;
80 dividersCount = Math.ceil((lastDividerTime - firstDividerTime) / gridSliceTi me);
81
82 if (!gridSliceTime)
83 dividersCount = 0;
84
85 var offsets = [];
86 for (var i = 0; i < dividersCount; ++i) {
87 var time = firstDividerTime + gridSliceTime * i;
88 if (calculator.computePosition(time) < freeZoneAtLeft)
89 continue;
90 offsets.push(time);
91 }
92
93 return {offsets: offsets, precision: Math.max(0, -Math.floor(Math.log(gridSl iceTime * 1.01) / Math.LN10))};
94 }
95
96 /**
97 * @param {!CanvasRenderingContext2D} context
98 * @param {!UI.TimelineGrid.Calculator} calculator
99 * @param {number} paddingTop
100 * @param {number=} headerHeight
101 * @param {number=} freeZoneAtLeft
102 */
103 static drawCanvasGrid(context, calculator, paddingTop, headerHeight, freeZoneA tLeft) {
104 context.save();
105 var ratio = window.devicePixelRatio;
106 context.scale(ratio, ratio);
107 var width = context.canvas.width / window.devicePixelRatio;
108 var height = context.canvas.height / window.devicePixelRatio;
109 var dividersData = UI.TimelineGrid.calculateDividerOffsets(calculator);
110 var dividerOffsets = dividersData.offsets;
111 var precision = dividersData.precision;
112
113 if (headerHeight) {
114 context.fillStyle = UI.themeSupport.patchColor('rgba(255, 255, 255, 0.5)', UI.ThemeSupport.ColorUsage.Background);
115 context.fillRect(0, 0, width, headerHeight);
116 }
117
118 context.fillStyle = UI.themeSupport.patchColor('#333', UI.ThemeSupport.Color Usage.Foreground);
119 context.strokeStyle = UI.themeSupport.patchColor('rgba(0, 0, 0, 0.1)', UI.Th emeSupport.ColorUsage.Foreground);
120 context.textBaseline = 'hanging';
121 context.font = '11px ' + Host.fontFamily();
122 context.lineWidth = 1;
123
124 context.translate(0.5, 0.5);
125 const paddingRight = 4;
126 for (var i = 0; i < dividerOffsets.length; ++i) {
127 var time = dividerOffsets[i];
128 var position = calculator.computePosition(time);
129 context.moveTo(position, 0);
130 context.lineTo(position, height);
131 if (!headerHeight)
132 continue;
133 var text = calculator.formatValue(time, precision);
134 var textWidth = context.measureText(text).width;
135 var textPosition = position - textWidth - paddingRight;
136 if (!freeZoneAtLeft || freeZoneAtLeft < textPosition)
137 context.fillText(text, textPosition, paddingTop);
138 }
139 context.stroke();
140 context.restore();
141 }
142
143 get dividersElement() {
144 return this._dividersElement;
145 }
146
147 get dividersLabelBarElement() {
148 return this._dividersLabelBarElement;
149 }
150
151 removeDividers() {
152 this._dividersElement.removeChildren();
153 this._dividersLabelBarElement.removeChildren();
154 }
155
156 /**
157 * @param {!UI.TimelineGrid.Calculator} calculator
158 * @param {number=} freeZoneAtLeft
159 * @return {boolean}
160 */
161 updateDividers(calculator, freeZoneAtLeft) {
162 var dividersData = UI.TimelineGrid.calculateDividerOffsets(calculator, freeZ oneAtLeft);
163 var dividerOffsets = dividersData.offsets;
164 var precision = dividersData.precision;
165
166 var dividersElementClientWidth = this._dividersElement.clientWidth;
167
168 // Reuse divider elements and labels.
169 var divider = /** @type {?Element} */ (this._dividersElement.firstChild);
170 var dividerLabelBar = /** @type {?Element} */ (this._dividersLabelBarElement .firstChild);
171
172 for (var i = 0; i < dividerOffsets.length; ++i) {
173 if (!divider) {
174 divider = createElement('div');
175 divider.className = 'resources-divider';
176 this._dividersElement.appendChild(divider);
177
178 dividerLabelBar = createElement('div');
179 dividerLabelBar.className = 'resources-divider';
180 var label = createElement('div');
181 label.className = 'resources-divider-label';
182 dividerLabelBar._labelElement = label;
183 dividerLabelBar.appendChild(label);
184 this._dividersLabelBarElement.appendChild(dividerLabelBar);
185 }
186
187 var time = dividerOffsets[i];
188 var position = calculator.computePosition(time);
189 dividerLabelBar._labelElement.textContent = calculator.formatValue(time, p recision);
190
191 var percentLeft = 100 * position / dividersElementClientWidth;
192 divider.style.left = percentLeft + '%';
193 dividerLabelBar.style.left = percentLeft + '%';
194
195 divider = /** @type {?Element} */ (divider.nextSibling);
196 dividerLabelBar = /** @type {?Element} */ (dividerLabelBar.nextSibling);
197 }
198
199 // Remove extras.
200 while (divider) {
201 var nextDivider = divider.nextSibling;
202 this._dividersElement.removeChild(divider);
203 divider = nextDivider;
204 }
205 while (dividerLabelBar) {
206 var nextDivider = dividerLabelBar.nextSibling;
207 this._dividersLabelBarElement.removeChild(dividerLabelBar);
208 dividerLabelBar = nextDivider;
209 }
210 return true;
211 }
212
213 /**
214 * @param {!Element} divider
215 */
216 addEventDivider(divider) {
217 this._eventDividersElement.appendChild(divider);
218 }
219
220 /**
221 * @param {!Array.<!Element>} dividers
222 */
223 addEventDividers(dividers) {
224 this._gridHeaderElement.removeChild(this._eventDividersElement);
225 for (var divider of dividers)
226 this._eventDividersElement.appendChild(divider);
227 this._gridHeaderElement.appendChild(this._eventDividersElement);
228 }
229
230 removeEventDividers() {
231 this._eventDividersElement.removeChildren();
232 }
233
234 hideEventDividers() {
235 this._eventDividersElement.classList.add('hidden');
236 }
237
238 showEventDividers() {
239 this._eventDividersElement.classList.remove('hidden');
240 }
241
242 hideDividers() {
243 this._dividersElement.classList.add('hidden');
244 }
245
246 showDividers() {
247 this._dividersElement.classList.remove('hidden');
248 }
249
250 /**
251 * @param {number} scrollTop
252 */
253 setScrollTop(scrollTop) {
254 this._dividersLabelBarElement.style.top = scrollTop + 'px';
255 this._eventDividersElement.style.top = scrollTop + 'px';
256 }
257 };
258
259
260 /**
261 * @interface
262 */
263 UI.TimelineGrid.Calculator = function() {};
264
265 UI.TimelineGrid.Calculator.prototype = {
266 /**
267 * @return {number}
268 */
269 paddingLeft() {},
270
271 /**
272 * @param {number} time
273 * @return {number}
274 */
275 computePosition(time) {},
276
277 /**
278 * @param {number} time
279 * @param {number=} precision
280 * @return {string}
281 */
282 formatValue(time, precision) {},
283
284 /** @return {number} */
285 minimumBoundary() {},
286
287 /** @return {number} */
288 zeroTime() {},
289
290 /** @return {number} */
291 maximumBoundary() {},
292
293 /** @return {number} */
294 boundarySpan() {}
295 };
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698