Added MacOS SDK
This commit is contained in:
@@ -0,0 +1,245 @@
|
||||
/*
|
||||
* Copyright (C) 2019 Apple Inc. All rights reserved.
|
||||
*
|
||||
* Redistribution and use in source and binary forms, with or without
|
||||
* modification, are permitted provided that the following conditions
|
||||
* are met:
|
||||
* 1. Redistributions of source code must retain the above copyright
|
||||
* notice, this list of conditions and the following disclaimer.
|
||||
* 2. Redistributions in binary form must reproduce the above copyright
|
||||
* notice, this list of conditions and the following disclaimer in the
|
||||
* documentation and/or other materials provided with the distribution.
|
||||
*
|
||||
* THIS SOFTWARE IS PROVIDED BY APPLE INC. AND ITS CONTRIBUTORS ``AS IS''
|
||||
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
|
||||
* THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
|
||||
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR ITS CONTRIBUTORS
|
||||
* BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
|
||||
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
|
||||
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
|
||||
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
|
||||
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
|
||||
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF
|
||||
* THE POSSIBILITY OF SUCH DAMAGE.
|
||||
*/
|
||||
|
||||
WI.CPUUsageCombinedView = class CPUUsageCombinedView extends WI.View
|
||||
{
|
||||
constructor(displayName)
|
||||
{
|
||||
super();
|
||||
|
||||
this.element.classList.add("cpu-usage-combined-view");
|
||||
|
||||
this._detailsElement = this.element.appendChild(document.createElement("div"));
|
||||
this._detailsElement.classList.add("details");
|
||||
|
||||
let detailsNameElement = this._detailsElement.appendChild(document.createElement("span"));
|
||||
detailsNameElement.classList.add("name");
|
||||
detailsNameElement.textContent = displayName;
|
||||
|
||||
this._detailsElement.appendChild(document.createElement("br"));
|
||||
this._detailsAverageElement = this._detailsElement.appendChild(document.createElement("span"));
|
||||
this._detailsElement.appendChild(document.createElement("br"));
|
||||
this._detailsMaxElement = this._detailsElement.appendChild(document.createElement("span"));
|
||||
this._detailsElement.appendChild(document.createElement("br"));
|
||||
this._detailsElement.appendChild(document.createElement("br"));
|
||||
this._updateDetails(NaN, NaN);
|
||||
|
||||
this._graphElement = this.element.appendChild(document.createElement("div"));
|
||||
this._graphElement.classList.add("graph");
|
||||
|
||||
// Combined thread usage area chart.
|
||||
this._chart = new WI.StackedAreaChart;
|
||||
this._chart.initializeSections(["main-thread-usage", "worker-thread-usage", "total-usage"]);
|
||||
this.addSubview(this._chart);
|
||||
this._graphElement.appendChild(this._chart.element);
|
||||
|
||||
// Main thread indicator strip.
|
||||
this._rangeChart = new WI.RangeChart;
|
||||
this.addSubview(this._rangeChart);
|
||||
this._graphElement.appendChild(this._rangeChart.element);
|
||||
|
||||
function appendLegendRow(legendElement, className) {
|
||||
let rowElement = legendElement.appendChild(document.createElement("div"));
|
||||
rowElement.classList.add("row");
|
||||
|
||||
let swatchElement = rowElement.appendChild(document.createElement("div"));
|
||||
swatchElement.classList.add("swatch", className);
|
||||
|
||||
let labelElement = rowElement.appendChild(document.createElement("div"));
|
||||
labelElement.classList.add("label");
|
||||
|
||||
return labelElement;
|
||||
}
|
||||
|
||||
this._legendElement = this._detailsElement.appendChild(document.createElement("div"));
|
||||
this._legendElement.classList.add("legend-container");
|
||||
|
||||
this._legendMainThreadElement = appendLegendRow(this._legendElement, "main-thread");
|
||||
this._legendWorkerThreadsElement = appendLegendRow(this._legendElement, "worker-threads");
|
||||
this._legendOtherThreadsElement = appendLegendRow(this._legendElement, "other-threads");
|
||||
this._legendTotalThreadsElement = appendLegendRow(this._legendElement, "total");
|
||||
|
||||
this.clearLegend();
|
||||
}
|
||||
|
||||
// Public
|
||||
|
||||
get graphElement() { return this._graphElement; }
|
||||
get chart() { return this._chart; }
|
||||
get rangeChart() { return this._rangeChart; }
|
||||
|
||||
clear()
|
||||
{
|
||||
this._cachedAverageSize = undefined;
|
||||
this._cachedMaxSize = undefined;
|
||||
this._updateDetails(NaN, NaN);
|
||||
|
||||
this.clearLegend();
|
||||
|
||||
this._chart.clear();
|
||||
this._chart.needsLayout();
|
||||
|
||||
this._rangeChart.clear();
|
||||
this._rangeChart.needsLayout();
|
||||
}
|
||||
|
||||
updateChart(dataPoints, size, visibleEndTime, min, max, average, xScale, yScale)
|
||||
{
|
||||
console.assert(size instanceof WI.Size);
|
||||
console.assert(min >= 0);
|
||||
console.assert(max >= 0);
|
||||
console.assert(min <= max);
|
||||
console.assert(min <= average && average <= max);
|
||||
|
||||
this._updateDetails(max, average);
|
||||
|
||||
this._chart.clearPoints();
|
||||
this._chart.size = size;
|
||||
this._chart.needsLayout();
|
||||
|
||||
if (!dataPoints.length)
|
||||
return;
|
||||
|
||||
// Ensure an empty graph is empty.
|
||||
if (!max)
|
||||
return;
|
||||
|
||||
// Extend the first data point to the start so it doesn't look like we originate at zero size.
|
||||
let firstX = 0;
|
||||
let firstY1 = yScale(dataPoints[0].mainThreadUsage);
|
||||
let firstY2 = yScale(dataPoints[0].mainThreadUsage + dataPoints[0].workerThreadUsage);
|
||||
let firstY3 = yScale(dataPoints[0].usage);
|
||||
this._chart.addPointSet(firstX, [firstY1, firstY2, firstY3]);
|
||||
|
||||
// Points for data points.
|
||||
for (let dataPoint of dataPoints) {
|
||||
let x = xScale(dataPoint.time);
|
||||
let y1 = yScale(dataPoint.mainThreadUsage);
|
||||
let y2 = yScale(dataPoint.mainThreadUsage + dataPoint.workerThreadUsage);
|
||||
let y3 = yScale(dataPoint.usage);
|
||||
this._chart.addPointSet(x, [y1, y2, y3]);
|
||||
}
|
||||
|
||||
// Extend the last data point to the end time.
|
||||
let lastDataPoint = dataPoints.lastValue;
|
||||
let lastX = Math.floor(xScale(visibleEndTime));
|
||||
let lastY1 = yScale(lastDataPoint.mainThreadUsage);
|
||||
let lastY2 = yScale(lastDataPoint.mainThreadUsage + lastDataPoint.workerThreadUsage);
|
||||
let lastY3 = yScale(lastDataPoint.usage);
|
||||
this._chart.addPointSet(lastX, [lastY1, lastY2, lastY3]);
|
||||
}
|
||||
|
||||
updateMainThreadIndicator(samples, size, visibleEndTime, xScale)
|
||||
{
|
||||
console.assert(size instanceof WI.Size);
|
||||
|
||||
this._rangeChart.clear();
|
||||
this._rangeChart.size = size;
|
||||
this._rangeChart.needsLayout();
|
||||
|
||||
if (!samples.length)
|
||||
return;
|
||||
|
||||
// Coalesce ranges of samples.
|
||||
let ranges = [];
|
||||
let currentRange = null;
|
||||
let currentSampleType = undefined;
|
||||
for (let i = 0; i < samples.length; ++i) {
|
||||
// Back to idle, close any current chunk.
|
||||
let type = samples[i];
|
||||
if (!type) {
|
||||
if (currentRange) {
|
||||
ranges.push(currentRange);
|
||||
currentRange = null;
|
||||
currentSampleType = undefined;
|
||||
}
|
||||
continue;
|
||||
}
|
||||
|
||||
// Expand existing chunk.
|
||||
if (type === currentSampleType) {
|
||||
currentRange.endIndex = i;
|
||||
continue;
|
||||
}
|
||||
|
||||
// If type changed, close current chunk.
|
||||
if (currentSampleType) {
|
||||
ranges.push(currentRange);
|
||||
currentRange = null;
|
||||
currentSampleType = undefined;
|
||||
}
|
||||
|
||||
// Start a new chunk.
|
||||
console.assert(!currentRange);
|
||||
console.assert(!currentSampleType);
|
||||
currentRange = {type, startIndex: i, endIndex: i};
|
||||
currentSampleType = type;
|
||||
}
|
||||
|
||||
for (let {type, startIndex, endIndex} of ranges) {
|
||||
let startX = xScale(startIndex);
|
||||
let endX = xScale(endIndex + 1);
|
||||
let width = endX - startX;
|
||||
this._rangeChart.addRange(startX, width, type);
|
||||
}
|
||||
}
|
||||
|
||||
clearLegend()
|
||||
{
|
||||
this._legendMainThreadElement.textContent = WI.UIString("Main Thread");
|
||||
this._legendWorkerThreadsElement.textContent = WI.UIString("Worker Threads");
|
||||
this._legendOtherThreadsElement.textContent = WI.UIString("Other Threads");
|
||||
this._legendTotalThreadsElement.textContent = "";
|
||||
}
|
||||
|
||||
updateLegend(record)
|
||||
{
|
||||
if (!record) {
|
||||
this.clearLegend();
|
||||
return;
|
||||
}
|
||||
|
||||
let {usage, mainThreadUsage, workerThreadUsage, webkitThreadUsage, unknownThreadUsage} = record;
|
||||
|
||||
this._legendMainThreadElement.textContent = WI.UIString("Main: %s").format(Number.percentageString(mainThreadUsage / 100));
|
||||
this._legendWorkerThreadsElement.textContent = WI.UIString("Worker: %s").format(Number.percentageString(workerThreadUsage / 100));
|
||||
this._legendOtherThreadsElement.textContent = WI.UIString("Other: %s").format(Number.percentageString((webkitThreadUsage + unknownThreadUsage) / 100));
|
||||
this._legendTotalThreadsElement.textContent = WI.UIString("Total: %s").format(Number.percentageString(usage / 100));
|
||||
}
|
||||
|
||||
// Private
|
||||
|
||||
_updateDetails(maxSize, averageSize)
|
||||
{
|
||||
if (this._cachedMaxSize === maxSize && this._cachedAverageSize === averageSize)
|
||||
return;
|
||||
|
||||
this._cachedAverageSize = averageSize;
|
||||
this._cachedMaxSize = maxSize;
|
||||
|
||||
this._detailsAverageElement.textContent = WI.UIString("Average: %s").format(Number.isFinite(maxSize) ? Number.percentageString(averageSize / 100) : emDash);
|
||||
this._detailsMaxElement.textContent = WI.UIString("Highest: %s").format(Number.isFinite(maxSize) ? Number.percentageString(maxSize / 100) : emDash);
|
||||
}
|
||||
};
|
||||
Reference in New Issue
Block a user