| /* |
| * Copyright (C) 2017 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.CanvasManager = class CanvasManager extends WI.Object |
| { |
| constructor() |
| { |
| super(); |
| |
| this._enabled = false; |
| this._canvasCollection = new WI.CanvasCollection; |
| this._canvasForIdentifierForTargetMap = new Map; |
| this._shaderProgramForIdentifierForTargetMap = new Map; |
| this._savedRecordings = new Set; |
| |
| WI.targetManager.addEventListener(WI.TargetManager.Event.TargetRemoved, this._handleTargetRemoved, this); |
| |
| WI.Frame.addEventListener(WI.Frame.Event.MainResourceDidChange, this._mainResourceDidChange, this); |
| } |
| |
| // Agent |
| |
| get domains() { return ["Canvas"]; } |
| |
| activateExtraDomain(domain) |
| { |
| // COMPATIBILITY (iOS 14.0): Inspector.activateExtraDomains was removed in favor of a declared debuggable type |
| |
| console.assert(domain === "Canvas"); |
| |
| for (let target of WI.targets) |
| this.initializeTarget(target); |
| } |
| |
| // Target |
| |
| initializeTarget(target) |
| { |
| if (!this._enabled) |
| return; |
| |
| if (target.hasDomain("Canvas")) { |
| target.CanvasAgent.enable(); |
| |
| if (WI.settings.canvasRecordingAutoCaptureEnabled.value && WI.settings.canvasRecordingAutoCaptureFrameCount.value) |
| target.CanvasAgent.setRecordingAutoCaptureFrameCount(WI.settings.canvasRecordingAutoCaptureFrameCount.value); |
| } |
| } |
| |
| // Public |
| |
| get canvasCollection() { return this._canvasCollection; } |
| get savedRecordings() { return this._savedRecordings; } |
| |
| async processJSON({filename, json, error}) |
| { |
| if (error) { |
| WI.Recording.synthesizeError(error); |
| return; |
| } |
| |
| if (typeof json !== "object" || json === null) { |
| WI.Recording.synthesizeError(WI.UIString("invalid JSON")); |
| return; |
| } |
| |
| let recording = WI.Recording.fromPayload(json); |
| if (!recording) |
| return; |
| |
| let extensionStart = filename.lastIndexOf("."); |
| if (extensionStart !== -1) |
| filename = filename.substring(0, extensionStart); |
| recording.createDisplayName(filename); |
| |
| this._savedRecordings.add(recording); |
| |
| this.dispatchEventToListeners(WI.CanvasManager.Event.RecordingSaved, {recording, imported: true, initiatedByUser: true}); |
| } |
| |
| enable() |
| { |
| console.assert(!this._enabled); |
| |
| this._enabled = true; |
| |
| for (let target of WI.targets) |
| this.initializeTarget(target); |
| } |
| |
| disable() |
| { |
| console.assert(this._enabled); |
| |
| for (let target of WI.targets) { |
| if (target.hasDomain("Canvas")) |
| target.CanvasAgent.disable(); |
| } |
| |
| this._canvasCollection.clear(); |
| this._canvasForIdentifierForTargetMap.clear(); |
| this._shaderProgramForIdentifierForTargetMap.clear(); |
| this._savedRecordings.clear(); |
| |
| this._enabled = false; |
| } |
| |
| setRecordingAutoCaptureFrameCount(enabled, count) |
| { |
| console.assert(!isNaN(count) && count >= 0); |
| |
| for (let target of WI.targets) { |
| if (target.hasDomain("Canvas")) |
| target.CanvasAgent.setRecordingAutoCaptureFrameCount(enabled ? count : 0); |
| } |
| |
| WI.settings.canvasRecordingAutoCaptureEnabled.value = enabled && count; |
| WI.settings.canvasRecordingAutoCaptureFrameCount.value = count; |
| } |
| |
| // CanvasObserver |
| |
| canvasAdded(target, canvasPayload) |
| { |
| let canvas = WI.Canvas.fromPayload(target, canvasPayload); |
| |
| let canvasForIdentifierMap = this._canvasForIdentifierForTargetMap.getOrInsert(target, new Map); |
| console.assert(!canvasForIdentifierMap.has(canvas.identifier), `Canvas already exists with id ${canvas.identifier}.`); |
| canvasForIdentifierMap.set(canvas.identifier, canvas); |
| |
| this._canvasCollection.add(canvas); |
| } |
| |
| canvasRemoved(target, canvasIdentifier) |
| { |
| let canvasForIdentifierMap = this._canvasForIdentifierForTargetMap.get(target); |
| console.assert(canvasForIdentifierMap); |
| if (!canvasForIdentifierMap) |
| return; |
| |
| let canvas = canvasForIdentifierMap.take(canvasIdentifier); |
| console.assert(canvas); |
| if (!canvas) |
| return; |
| |
| this._saveRecordings(canvas); |
| |
| this._canvasCollection.remove(canvas); |
| |
| let shaderProgramForIdentifierMap = this._shaderProgramForIdentifierForTargetMap.get(target); |
| if (shaderProgramForIdentifierMap) { |
| for (let program of canvas.shaderProgramCollection) |
| shaderProgramForIdentifierMap.delete(program.identifier); |
| } |
| |
| canvas.shaderProgramCollection.clear(); |
| } |
| |
| canvasSizeChanged(target, canvasIdentifier, width, height) |
| { |
| let canvas = this._canvasForIdentifier(target, canvasIdentifier); |
| if (!canvas) |
| return; |
| |
| canvas.sizeChanged(new WI.Size(width, height)); |
| } |
| |
| canvasMemoryChanged(target, canvasIdentifier, memoryCost) |
| { |
| let canvas = this._canvasForIdentifier(target, canvasIdentifier); |
| if (!canvas) |
| return; |
| |
| canvas.memoryChanged(memoryCost); |
| } |
| |
| clientNodesChanged(target, canvasIdentifier) |
| { |
| let canvas = this._canvasForIdentifier(target, canvasIdentifier); |
| if (!canvas) |
| return; |
| |
| canvas.clientNodesChanged(); |
| } |
| |
| recordingStarted(target, canvasIdentifier, initiator) |
| { |
| let canvas = this._canvasForIdentifier(target, canvasIdentifier); |
| if (!canvas) |
| return; |
| |
| canvas.recordingStarted(initiator); |
| } |
| |
| recordingProgress(target, canvasIdentifier, framesPayload, bufferUsed) |
| { |
| let canvas = this._canvasForIdentifier(target, canvasIdentifier); |
| if (!canvas) |
| return; |
| |
| canvas.recordingProgress(framesPayload, bufferUsed); |
| } |
| |
| recordingFinished(target, canvasIdentifier, recordingPayload) |
| { |
| let canvas = this._canvasForIdentifier(target, canvasIdentifier); |
| if (!canvas) |
| return; |
| |
| canvas.recordingFinished(recordingPayload); |
| } |
| |
| extensionEnabled(target, canvasIdentifier, extension) |
| { |
| let canvas = this._canvasForIdentifier(target, canvasIdentifier); |
| if (!canvas) |
| return; |
| |
| canvas.enableExtension(extension); |
| } |
| |
| programCreated(target, shaderProgramPayload) |
| { |
| let canvas = this._canvasForIdentifier(target, shaderProgramPayload.canvasId); |
| if (!canvas) |
| return; |
| |
| // COMPATIBILITY (iOS 13.0): `Canvas.ShaderProgram.programType` did not exist yet. |
| let programType = shaderProgramPayload.programType; |
| if (!programType) |
| programType = WI.ShaderProgram.ProgramType.Render; |
| |
| let options = {}; |
| |
| // COMPATIBILITY (iOS 13.0): `Canvas.ShaderProgram.sharesVertexFragmentShader` did not exist yet. |
| if (shaderProgramPayload.sharesVertexFragmentShader) |
| options.sharesVertexFragmentShader = true; |
| |
| let program = new WI.ShaderProgram(target, shaderProgramPayload.programId, programType, canvas, options); |
| |
| let shaderProgramForIdentifierMap = this._shaderProgramForIdentifierForTargetMap.getOrInsert(target, new Map); |
| console.assert(!shaderProgramForIdentifierMap.has(program.identifier), `ShaderProgram already exists with id ${program.identifier}.`); |
| shaderProgramForIdentifierMap.set(program.identifier, program); |
| |
| canvas.shaderProgramCollection.add(program); |
| } |
| |
| programDeleted(target, programIdentifier) |
| { |
| let shaderProgramForIdentifierMap = this._shaderProgramForIdentifierForTargetMap.get(target); |
| console.assert(shaderProgramForIdentifierMap); |
| if (!shaderProgramForIdentifierMap) |
| return; |
| |
| let program = shaderProgramForIdentifierMap.take(programIdentifier); |
| console.assert(program); |
| if (!program) |
| return; |
| |
| program.canvas.shaderProgramCollection.remove(program); |
| } |
| |
| // Private |
| |
| _canvasForIdentifier(target, canvasIdentifier) |
| { |
| let canvasForIdentifierMap = this._canvasForIdentifierForTargetMap.get(target); |
| console.assert(canvasForIdentifierMap); |
| if (!canvasForIdentifierMap) |
| return null; |
| |
| let canvas = canvasForIdentifierMap.get(canvasIdentifier); |
| console.assert(canvas); |
| if (!canvas) |
| return null; |
| |
| return canvas; |
| } |
| |
| _saveRecordings(canvas) |
| { |
| for (let recording of canvas.recordingCollection) { |
| recording.source = null; |
| recording.createDisplayName(recording.displayName); |
| this._savedRecordings.add(recording); |
| this.dispatchEventToListeners(WI.CanvasManager.Event.RecordingSaved, {recording}); |
| } |
| } |
| |
| _handleTargetRemoved(event) |
| { |
| let {target} = event.data; |
| |
| this._canvasForIdentifierForTargetMap.delete(target); |
| this._shaderProgramForIdentifierForTargetMap.delete(target); |
| } |
| |
| _mainResourceDidChange(event) |
| { |
| console.assert(event.target instanceof WI.Frame); |
| if (!event.target.isMainFrame()) |
| return; |
| |
| WI.Canvas.resetUniqueDisplayNameNumbers(); |
| |
| for (let canvasForIdentifierMap of this._canvasForIdentifierForTargetMap.values()) { |
| for (let canvas of canvasForIdentifierMap.values()) |
| this._saveRecordings(canvas); |
| } |
| |
| this._canvasCollection.clear(); |
| this._canvasForIdentifierForTargetMap.clear(); |
| this._shaderProgramForIdentifierForTargetMap.clear(); |
| } |
| |
| // Testing |
| |
| get shaderPrograms() |
| { |
| let programs = []; |
| for (let shaderProgramForIdentifierMap of this._shaderProgramForIdentifierForTargetMap.values()) |
| programs.pushAll(shaderProgramForIdentifierMap.values()); |
| return programs; |
| } |
| }; |
| |
| WI.CanvasManager.Event = { |
| RecordingSaved: "canvas-manager-recording-saved", |
| }; |