| /* |
| * Copyright (C) 2022 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.ScreenshotsTimelineOverviewGraph = class ScreenshotsTimelineOverviewGraph extends WI.TimelineOverviewGraph |
| { |
| constructor(timeline, timelineOverview) |
| { |
| console.assert(timeline instanceof WI.Timeline); |
| console.assert(timeline.type === WI.TimelineRecord.Type.Screenshots, timeline); |
| |
| super(timelineOverview); |
| |
| this.element.classList.add("screenshots"); |
| |
| this._screenshotsTimeline = timeline; |
| |
| this._lastSelectedRecordInLayout = null; |
| |
| this.reset(); |
| } |
| |
| // Protected |
| |
| get height() |
| { |
| return 60; |
| } |
| |
| reset() |
| { |
| super.reset(); |
| |
| this._imageElementForRecord = new WeakMap; |
| } |
| |
| layout() |
| { |
| super.layout(); |
| |
| if (this.hidden) |
| return; |
| |
| this.element.removeChildren(); |
| |
| let secondsPerPixel = this.timelineOverview.secondsPerPixel; |
| |
| for (let record of this._visibleRecords()) { |
| this.element.appendChild(this._imageElementForRecord.getOrInitialize(record, () => { |
| let imageElement = document.createElement("img"); |
| imageElement.height = this.height; |
| imageElement.style.left = (record.startTime - this.startTime) / secondsPerPixel + "px"; |
| |
| imageElement.hidden = true; |
| imageElement.addEventListener("load", (event) => { |
| imageElement.hidden = false; |
| }, {once: true}); |
| imageElement.src = record.imageData; |
| |
| imageElement.addEventListener("click", (event) => { |
| // Ensure that the container "click" listener added by `WI.TimelineOverview` isn't called. |
| event.__timelineRecordClickEventHandled = true; |
| |
| this.selectedRecord = record; |
| }); |
| |
| return imageElement; |
| })); |
| } |
| |
| if (this._lastSelectedRecordInLayout) |
| this._imageElementForRecord.get(this._lastSelectedRecordInLayout)?.classList.remove("selected"); |
| |
| this._lastSelectedRecordInLayout = this.selectedRecord; |
| |
| if (this._lastSelectedRecordInLayout) |
| this._imageElementForRecord.get(this._lastSelectedRecordInLayout)?.classList.add("selected"); |
| } |
| |
| updateSelectedRecord() |
| { |
| super.updateSelectedRecord(); |
| |
| if (this._lastSelectedRecordInLayout !== this.selectedRecord) { |
| // Since we don't have the exact element to re-style with a selected appearance |
| // we trigger another layout to re-layout the graph and provide additional |
| // styles for the column for the selected record. |
| this.needsLayout(); |
| } |
| } |
| |
| // Private |
| |
| _visibleRecords() |
| { |
| let visibleEndTime = Math.min(this.endTime, this.currentTime); |
| |
| let records = this._screenshotsTimeline.records; |
| let visibleRecords = []; |
| |
| for (let i = 0; i < records.length; ++i) { |
| let record = records[i]; |
| if (!(record.startTime >= this.startTime || record.startTime <= visibleEndTime)) |
| continue; |
| |
| if (!visibleRecords.length && i) |
| visibleRecords.push(records[i - 1]); |
| |
| visibleRecords.push(record); |
| } |
| |
| return visibleRecords; |
| } |
| }; |