UNPKG

@quick-game/cli

Version:

Command line interface for rapid qg development

1,139 lines 70.2 kB
/* * Copyright (C) 2012 Google 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: * * * Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * * 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. * * Neither the name of Google Inc. nor the names of its * contributors may be used to endorse or promote products derived from * this software without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND 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 THE COPYRIGHT * OWNER OR 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. */ import * as Common from '../../core/common/common.js'; import * as Host from '../../core/host/host.js'; import * as i18n from '../../core/i18n/i18n.js'; import * as Platform from '../../core/platform/platform.js'; import * as Root from '../../core/root/root.js'; import * as SDK from '../../core/sdk/sdk.js'; import * as Bindings from '../../models/bindings/bindings.js'; import * as Persistence from '../../models/persistence/persistence.js'; import * as Workspace from '../../models/workspace/workspace.js'; import * as IconButton from '../../ui/components/icon_button/icon_button.js'; import * as UI from '../../ui/legacy/legacy.js'; import * as Snippets from '../snippets/snippets.js'; import navigatorTreeStyles from './navigatorTree.css.js'; import navigatorViewStyles from './navigatorView.css.js'; import { SearchSourcesView } from './SearchSourcesView.js'; const UIStrings = { /** *@description Text in Navigator View of the Sources panel */ searchInFolder: 'Search in folder', /** *@description Search label in Navigator View of the Sources panel */ searchInAllFiles: 'Search in all files', /** *@description Text in Navigator View of the Sources panel */ noDomain: '(no domain)', /** *@description Text in Navigator View of the Sources panel */ authored: 'Authored', /** *@description Text in Navigator View of the Sources panel */ authoredTooltip: 'Contains original sources', /** *@description Text in Navigator View of the Sources panel */ deployed: 'Deployed', /** *@description Text in Navigator View of the Sources panel */ deployedTooltip: 'Contains final sources the browser sees', /** *@description Text in Navigator View of the Sources panel */ areYouSureYouWantToExcludeThis: 'Are you sure you want to exclude this folder?', /** *@description Text in Navigator View of the Sources panel */ areYouSureYouWantToDeleteThis: 'Are you sure you want to delete this file?', /** *@description A context menu item in the Navigator View of the Sources panel */ rename: 'Rename…', /** *@description A context menu item in the Navigator View of the Sources panel */ makeACopy: 'Make a copy…', /** *@description Text to delete something */ delete: 'Delete', /** *@description A button text to confirm an action to remove a folder. This is not the same as delete. It removes the folder from UI but do not delete them. */ remove: 'Remove', /** *@description Text in Navigator View of the Sources panel */ areYouSureYouWantToDeleteAll: 'Are you sure you want to delete all overrides in this folder?', /** *@description A context menu item in the Navigator View of the Sources panel */ openFolder: 'Open folder', /** *@description A context menu item in the Navigator View of the Sources panel */ newFile: 'New file', /** *@description A context menu item in the Navigator View of the Sources panel to exclude a folder from workspace */ excludeFolder: 'Exclude from workspace', /** *@description A context menu item in the Navigator View of the Sources panel */ removeFolderFromWorkspace: 'Remove from workspace', /** *@description Text in Navigator View of the Sources panel * @example {a-folder-name} PH1 */ areYouSureYouWantToRemoveThis: 'Remove ‘{PH1}’ from Workspace?', /** *@description Text in Navigator View of the Sources panel. Warning message when user remove a folder. */ workspaceStopSyncing: 'This will stop syncing changes from DevTools to your sources.', /** *@description A context menu item in the Navigator View of the Sources panel */ deleteAllOverrides: 'Delete all overrides', /** *@description Name of an item from source map *@example {compile.html} PH1 */ sFromSourceMap: '{PH1} (from source map)', /** *@description Name of an item that is on the ignore list *@example {compile.html} PH1 */ sIgnoreListed: '{PH1} (ignore listed)', }; const str_ = i18n.i18n.registerUIStrings('panels/sources/NavigatorView.ts', UIStrings); const i18nString = i18n.i18n.getLocalizedString.bind(undefined, str_); export const Types = { Authored: 'authored', Deployed: 'deployed', Domain: 'domain', File: 'file', FileSystem: 'fs', FileSystemFolder: 'fs-folder', Frame: 'frame', NetworkFolder: 'nw-folder', Root: 'root', Worker: 'worker', }; const TYPE_ORDERS = new Map([ [Types.Root, 1], [Types.Authored, 1], [Types.Deployed, 5], [Types.Domain, 10], [Types.FileSystemFolder, 1], [Types.NetworkFolder, 1], [Types.File, 10], [Types.Frame, 70], [Types.Worker, 90], [Types.FileSystem, 100], ]); export class NavigatorView extends UI.Widget.VBox { placeholder; scriptsTree; uiSourceCodeNodes; subfolderNodes; rootNode; frameNodes; authoredNode; deployedNode; // TODO(crbug.com/1172300) Ignored during the jsdoc to ts migration) // eslint-disable-next-line @typescript-eslint/no-explicit-any navigatorGroupByFolderSetting; navigatorGroupByAuthoredExperiment; workspaceInternal; lastSelectedUISourceCode; groupByFrame; groupByAuthored; // TODO(crbug.com/1172300) Ignored during the jsdoc to ts migration) // eslint-disable-next-line @typescript-eslint/no-explicit-any groupByDomain; // TODO(crbug.com/1172300) Ignored during the jsdoc to ts migration) // eslint-disable-next-line @typescript-eslint/no-explicit-any groupByFolder; constructor(enableAuthoredGrouping) { super(true); this.placeholder = null; this.scriptsTree = new UI.TreeOutline.TreeOutlineInShadow(); this.scriptsTree.setComparator(NavigatorView.treeElementsCompare); this.scriptsTree.setFocusable(false); this.contentElement.appendChild(this.scriptsTree.element); this.setDefaultFocusedElement(this.scriptsTree.element); this.uiSourceCodeNodes = new Platform.MapUtilities.Multimap(); this.subfolderNodes = new Map(); this.rootNode = new NavigatorRootTreeNode(this); this.rootNode.populate(); this.frameNodes = new Map(); this.contentElement.addEventListener('contextmenu', this.handleContextMenu.bind(this), false); UI.ShortcutRegistry.ShortcutRegistry.instance().addShortcutListener(this.contentElement, { 'sources.rename': this.renameShortcut.bind(this) }); this.navigatorGroupByFolderSetting = Common.Settings.Settings.instance().moduleSetting('navigatorGroupByFolder'); this.navigatorGroupByFolderSetting.addChangeListener(this.groupingChanged.bind(this)); if (enableAuthoredGrouping) { this.navigatorGroupByAuthoredExperiment = Root.Runtime.ExperimentName.AUTHORED_DEPLOYED_GROUPING; } Bindings.IgnoreListManager.IgnoreListManager.instance().addChangeListener(this.ignoreListChanged.bind(this)); this.initGrouping(); Persistence.Persistence.PersistenceImpl.instance().addEventListener(Persistence.Persistence.Events.BindingCreated, this.onBindingChanged, this); Persistence.Persistence.PersistenceImpl.instance().addEventListener(Persistence.Persistence.Events.BindingRemoved, this.onBindingChanged, this); Persistence.NetworkPersistenceManager.NetworkPersistenceManager.instance().addEventListener(Persistence.NetworkPersistenceManager.Events.RequestsForHeaderOverridesFileChanged, this.#onRequestsForHeaderOverridesFileChanged, this); SDK.TargetManager.TargetManager.instance().addEventListener(SDK.TargetManager.Events.NameChanged, this.targetNameChanged, this); SDK.TargetManager.TargetManager.instance().observeTargets(this); this.resetWorkspace(Workspace.Workspace.WorkspaceImpl.instance()); this.workspaceInternal.uiSourceCodes().forEach(this.addUISourceCode.bind(this)); Bindings.NetworkProject.NetworkProjectManager.instance().addEventListener(Bindings.NetworkProject.Events.FrameAttributionAdded, this.frameAttributionAdded, this); Bindings.NetworkProject.NetworkProjectManager.instance().addEventListener(Bindings.NetworkProject.Events.FrameAttributionRemoved, this.frameAttributionRemoved, this); } static treeElementOrder(treeElement) { if (boostOrderForNode.has(treeElement)) { return 0; } const actualElement = treeElement; let order = TYPE_ORDERS.get(actualElement.nodeType) || 0; if (actualElement.uiSourceCode) { const contentType = actualElement.uiSourceCode.contentType(); if (contentType.isDocument()) { order += 3; } else if (contentType.isScript()) { order += 5; } else if (contentType.isStyleSheet()) { order += 10; } else { order += 15; } } return order; } static appendSearchItem(contextMenu, path) { let searchLabel = i18nString(UIStrings.searchInFolder); if (!path || !path.trim()) { path = '*'; searchLabel = i18nString(UIStrings.searchInAllFiles); } contextMenu.viewSection().appendItem(searchLabel, () => { if (path) { void SearchSourcesView.openSearch(`file:${path.trim()}`); } }); } static treeElementsCompare(treeElement1, treeElement2) { const typeWeight1 = NavigatorView.treeElementOrder(treeElement1); const typeWeight2 = NavigatorView.treeElementOrder(treeElement2); if (typeWeight1 > typeWeight2) { return 1; } if (typeWeight1 < typeWeight2) { return -1; } return Platform.StringUtilities.naturalOrderComparator(treeElement1.titleAsText(), treeElement2.titleAsText()); } setPlaceholder(placeholder) { console.assert(!this.placeholder, 'A placeholder widget was already set'); this.placeholder = placeholder; placeholder.show(this.contentElement, this.contentElement.firstChild); updateVisibility.call(this); this.scriptsTree.addEventListener(UI.TreeOutline.Events.ElementAttached, updateVisibility.bind(this)); this.scriptsTree.addEventListener(UI.TreeOutline.Events.ElementsDetached, updateVisibility.bind(this)); function updateVisibility() { const showTree = this.scriptsTree.firstChild(); if (showTree) { placeholder.hideWidget(); } else { placeholder.showWidget(); } this.scriptsTree.element.classList.toggle('hidden', !showTree); } } onBindingChanged(event) { const binding = event.data; let isFromSourceMap = false; // Update UISourceCode titles. const networkNodes = this.uiSourceCodeNodes.get(binding.network); for (const networkNode of networkNodes) { networkNode.updateTitle(); isFromSourceMap ||= networkNode.uiSourceCode().contentType().isFromSourceMap(); } const fileSystemNodes = this.uiSourceCodeNodes.get(binding.fileSystem); for (const fileSystemNode of fileSystemNodes) { fileSystemNode.updateTitle(); isFromSourceMap ||= fileSystemNode.uiSourceCode().contentType().isFromSourceMap(); } // Update folder titles. const pathTokens = Persistence.FileSystemWorkspaceBinding.FileSystemWorkspaceBinding.relativePath(binding.fileSystem); let folderPath = Platform.DevToolsPath.EmptyEncodedPathString; for (let i = 0; i < pathTokens.length - 1; ++i) { folderPath = Common.ParsedURL.ParsedURL.concatenate(folderPath, pathTokens[i]); const folderId = this.folderNodeId(binding.fileSystem.project(), null, null, binding.fileSystem.origin(), isFromSourceMap, folderPath); const folderNode = this.subfolderNodes.get(folderId); if (folderNode) { folderNode.updateTitle(); } folderPath = Common.ParsedURL.ParsedURL.concatenate(folderPath, '/'); } // Update fileSystem root title. const fileSystemRoot = this.rootOrDeployedNode().child(binding.fileSystem.project().id()); if (fileSystemRoot) { fileSystemRoot.updateTitle(); } } #onRequestsForHeaderOverridesFileChanged(event) { const headersFileUiSourceCode = event.data; const networkNodes = this.uiSourceCodeNodes.get(headersFileUiSourceCode); for (const networkNode of networkNodes) { networkNode.updateTitle(); } } focus() { this.scriptsTree.focus(); } /** * Central place to add elements to the tree to * enable focus if the tree has elements */ appendChild(parent, child) { this.scriptsTree.setFocusable(true); parent.appendChild(child); } /** * Central place to remove elements from the tree to * disable focus if the tree is empty */ removeChild(parent, child) { parent.removeChild(child); if (this.scriptsTree.rootElement().childCount() === 0) { this.scriptsTree.setFocusable(false); } } resetWorkspace(workspace) { // Clear old event listeners first. if (this.workspaceInternal) { this.workspaceInternal.removeEventListener(Workspace.Workspace.Events.UISourceCodeAdded, this.uiSourceCodeAddedCallback, this); this.workspaceInternal.removeEventListener(Workspace.Workspace.Events.UISourceCodeRemoved, this.uiSourceCodeRemovedCallback, this); this.workspaceInternal.removeEventListener(Workspace.Workspace.Events.ProjectAdded, this.projectAddedCallback, this); this.workspaceInternal.removeEventListener(Workspace.Workspace.Events.ProjectRemoved, this.projectRemovedCallback, this); } this.workspaceInternal = workspace; this.workspaceInternal.addEventListener(Workspace.Workspace.Events.UISourceCodeAdded, this.uiSourceCodeAddedCallback, this); this.workspaceInternal.addEventListener(Workspace.Workspace.Events.UISourceCodeRemoved, this.uiSourceCodeRemovedCallback, this); this.workspaceInternal.addEventListener(Workspace.Workspace.Events.ProjectAdded, this.projectAddedCallback, this); this.workspaceInternal.addEventListener(Workspace.Workspace.Events.ProjectRemoved, this.projectRemovedCallback, this); this.workspaceInternal.projects().forEach(this.projectAdded.bind(this)); this.computeUniqueFileSystemProjectNames(); } projectAddedCallback(event) { const project = event.data; this.projectAdded(project); if (project.type() === Workspace.Workspace.projectTypes.FileSystem) { this.computeUniqueFileSystemProjectNames(); } } projectRemovedCallback(event) { const project = event.data; this.removeProject(project); if (project.type() === Workspace.Workspace.projectTypes.FileSystem) { this.computeUniqueFileSystemProjectNames(); } } workspace() { return this.workspaceInternal; } acceptProject(project) { return !project.isServiceProject(); } frameAttributionAdded(event) { const { uiSourceCode } = event.data; if (!this.acceptsUISourceCode(uiSourceCode)) { return; } const addedFrame = event.data.frame; // This event does not happen for UISourceCodes without initial attribution. this.addUISourceCodeNode(uiSourceCode, addedFrame); } frameAttributionRemoved(event) { const { uiSourceCode } = event.data; if (!this.acceptsUISourceCode(uiSourceCode)) { return; } const removedFrame = event.data.frame; const node = Array.from(this.uiSourceCodeNodes.get(uiSourceCode)).find(node => node.frame() === removedFrame); if (node) { this.removeUISourceCodeNode(node); } } acceptsUISourceCode(uiSourceCode) { return this.acceptProject(uiSourceCode.project()); } addUISourceCode(uiSourceCode) { if (Root.Runtime.experiments.isEnabled(Root.Runtime.ExperimentName.JUST_MY_CODE) && Bindings.IgnoreListManager.IgnoreListManager.instance().isUserOrSourceMapIgnoreListedUISourceCode(uiSourceCode)) { return; } if (!this.acceptsUISourceCode(uiSourceCode)) { return; } if (uiSourceCode.isFetchXHR()) { return; } const frames = Bindings.NetworkProject.NetworkProject.framesForUISourceCode(uiSourceCode); if (frames.length) { for (const frame of frames) { this.addUISourceCodeNode(uiSourceCode, frame); } } else { this.addUISourceCodeNode(uiSourceCode, null); } this.uiSourceCodeAdded(uiSourceCode); } addUISourceCodeNode(uiSourceCode, frame) { const isFromSourceMap = uiSourceCode.contentType().isFromSourceMap(); let path; if (uiSourceCode.project().type() === Workspace.Workspace.projectTypes.FileSystem) { path = Persistence.FileSystemWorkspaceBinding.FileSystemWorkspaceBinding.relativePath(uiSourceCode).slice(0, -1); } else { path = Common.ParsedURL.ParsedURL.extractPath(uiSourceCode.url()).split('/').slice(1, -1); } const project = uiSourceCode.project(); const target = Bindings.NetworkProject.NetworkProject.targetForUISourceCode(uiSourceCode); const folderNode = this.folderNode(uiSourceCode, project, target, frame, uiSourceCode.origin(), path, isFromSourceMap); const uiSourceCodeNode = new NavigatorUISourceCodeTreeNode(this, uiSourceCode, frame); const existingNode = folderNode.child(uiSourceCodeNode.id); if (existingNode && existingNode instanceof NavigatorUISourceCodeTreeNode) { this.uiSourceCodeNodes.set(uiSourceCode, existingNode); } else { folderNode.appendChild(uiSourceCodeNode); this.uiSourceCodeNodes.set(uiSourceCode, uiSourceCodeNode); uiSourceCodeNode.updateTitleBubbleUp(); } this.selectDefaultTreeNode(); } uiSourceCodeAdded(_uiSourceCode) { } uiSourceCodeAddedCallback(event) { const uiSourceCode = event.data; this.addUISourceCode(uiSourceCode); } uiSourceCodeRemovedCallback(event) { this.removeUISourceCodes([event.data]); } tryAddProject(project) { this.projectAdded(project); for (const uiSourceCode of project.uiSourceCodes()) { this.addUISourceCode(uiSourceCode); } } projectAdded(project) { const rootOrDeployed = this.rootOrDeployedNode(); if (!this.acceptProject(project) || project.type() !== Workspace.Workspace.projectTypes.FileSystem || Snippets.ScriptSnippetFileSystem.isSnippetsProject(project) || rootOrDeployed.child(project.id())) { return; } rootOrDeployed.appendChild(new NavigatorGroupTreeNode(this, project, project.id(), Types.FileSystem, project.displayName())); this.selectDefaultTreeNode(); } // TODO(einbinder) remove this code after crbug.com/964075 is fixed selectDefaultTreeNode() { const children = this.rootNode.children(); if (children.length && !this.scriptsTree.selectedTreeElement) { children[0].treeNode().select(true /* omitFocus */, false /* selectedByUser */); } } computeUniqueFileSystemProjectNames() { const fileSystemProjects = this.workspaceInternal.projectsForType(Workspace.Workspace.projectTypes.FileSystem); if (!fileSystemProjects.length) { return; } const reversedIndex = Common.Trie.Trie.newArrayTrie(); const reversedPaths = []; for (const project of fileSystemProjects) { const fileSystem = project; const reversedPathParts = fileSystem.fileSystemPath().split('/').reverse(); reversedPaths.push(reversedPathParts); reversedIndex.add(reversedPathParts); } const rootOrDeployed = this.rootOrDeployedNode(); for (let i = 0; i < fileSystemProjects.length; ++i) { const reversedPath = reversedPaths[i]; const project = fileSystemProjects[i]; reversedIndex.remove(reversedPath); const commonPrefix = reversedIndex.longestPrefix(reversedPath, false /* fullWordOnly */); reversedIndex.add(reversedPath); const prefixPath = reversedPath.slice(0, commonPrefix.length + 1); const path = Common.ParsedURL.ParsedURL.encodedPathToRawPathString(prefixPath.reverse().join('/')); const fileSystemNode = rootOrDeployed.child(project.id()); if (fileSystemNode) { fileSystemNode.setTitle(path); } } } removeProject(project) { this.removeUISourceCodes(project.uiSourceCodes()); if (project.type() !== Workspace.Workspace.projectTypes.FileSystem) { return; } const fileSystemNode = this.rootNode.child(project.id()); if (!fileSystemNode) { return; } this.rootNode.removeChild(fileSystemNode); } folderNodeId(project, target, frame, projectOrigin, isFromSourceMap, path) { const projectId = project.type() === Workspace.Workspace.projectTypes.FileSystem ? project.id() : ''; let targetId = target && !(this.groupByAuthored && isFromSourceMap) ? target.id() : ''; let frameId = this.groupByFrame && frame ? frame.id : ''; if (this.groupByAuthored) { if (isFromSourceMap) { targetId = 'Authored'; frameId = ''; } else { targetId = 'Deployed:' + targetId; } } return targetId + ':' + projectId + ':' + frameId + ':' + projectOrigin + ':' + path; } folderNode(uiSourceCode, project, target, frame, projectOrigin, path, fromSourceMap) { if (Snippets.ScriptSnippetFileSystem.isSnippetsUISourceCode(uiSourceCode)) { return this.rootNode; } if (target && !this.groupByFolder && !fromSourceMap) { return this.domainNode(uiSourceCode, project, target, frame, projectOrigin); } const folderPath = Common.ParsedURL.ParsedURL.join(path, '/'); const folderId = this.folderNodeId(project, target, frame, projectOrigin, fromSourceMap, folderPath); let folderNode = this.subfolderNodes.get(folderId); if (folderNode) { return folderNode; } if (!path.length) { if (target) { return this.domainNode(uiSourceCode, project, target, frame, projectOrigin); } return this.rootOrDeployedNode().child(project.id()); } const parentNode = this.folderNode(uiSourceCode, project, target, frame, projectOrigin, path.slice(0, -1), fromSourceMap); let type = Types.NetworkFolder; if (project.type() === Workspace.Workspace.projectTypes.FileSystem) { type = Types.FileSystemFolder; } const name = Common.ParsedURL.ParsedURL.encodedPathToRawPathString(path[path.length - 1]); folderNode = new NavigatorFolderTreeNode(this, project, folderId, type, folderPath, name, projectOrigin); this.subfolderNodes.set(folderId, folderNode); parentNode.appendChild(folderNode); return folderNode; } domainNode(uiSourceCode, project, target, frame, projectOrigin) { const isAuthored = uiSourceCode.contentType().isFromSourceMap(); const frameNode = this.frameNode(project, target, frame, isAuthored); if (!this.groupByDomain) { return frameNode; } let domainNode = frameNode.child(projectOrigin); if (domainNode) { return domainNode; } domainNode = new NavigatorGroupTreeNode(this, project, projectOrigin, Types.Domain, this.computeProjectDisplayName(target, projectOrigin)); if (frame && projectOrigin === Common.ParsedURL.ParsedURL.extractOrigin(frame.url)) { boostOrderForNode.add(domainNode.treeNode()); } frameNode.appendChild(domainNode); if (isAuthored && this.groupByAuthored) { domainNode.treeNode().expand(); } return domainNode; } frameNode(project, target, frame, isAuthored) { if (!this.groupByFrame || !frame || (this.groupByAuthored && isAuthored)) { return this.targetNode(project, target, isAuthored); } let frameNode = this.frameNodes.get(frame); if (frameNode) { return frameNode; } frameNode = new NavigatorGroupTreeNode(this, project, target.id() + ':' + frame.id, Types.Frame, frame.displayName()); frameNode.setHoverCallback(hoverCallback); this.frameNodes.set(frame, frameNode); const parentFrame = frame.parentFrame(); this.frameNode(project, parentFrame ? parentFrame.resourceTreeModel().target() : target, parentFrame, isAuthored) .appendChild(frameNode); if (!parentFrame) { boostOrderForNode.add(frameNode.treeNode()); frameNode.treeNode().expand(); } function hoverCallback(hovered) { if (hovered) { const overlayModel = target.model(SDK.OverlayModel.OverlayModel); if (overlayModel && frame) { overlayModel.highlightFrame(frame.id); } } else { SDK.OverlayModel.OverlayModel.hideDOMNodeHighlight(); } } return frameNode; } targetNode(project, target, isAuthored) { if (this.groupByAuthored && isAuthored) { if (!this.authoredNode) { this.authoredNode = new NavigatorGroupTreeNode(this, null, 'group:Authored', Types.Authored, i18nString(UIStrings.authored), i18nString(UIStrings.authoredTooltip)); this.rootNode.appendChild(this.authoredNode); this.authoredNode.treeNode().expand(); } return this.authoredNode; } const rootOrDeployed = this.rootOrDeployedNode(); if (target === SDK.TargetManager.TargetManager.instance().scopeTarget()) { return rootOrDeployed; } let targetNode = rootOrDeployed.child('target:' + target.id()); if (!targetNode) { targetNode = new NavigatorGroupTreeNode(this, project, 'target:' + target.id(), target.type() === SDK.Target.Type.Frame ? Types.Frame : Types.Worker, target.name()); rootOrDeployed.appendChild(targetNode); } return targetNode; } rootOrDeployedNode() { if (this.groupByAuthored) { if (!this.deployedNode) { this.deployedNode = new NavigatorGroupTreeNode(this, null, 'group:Deployed', Types.Deployed, i18nString(UIStrings.deployed), i18nString(UIStrings.deployedTooltip)); this.rootNode.appendChild(this.deployedNode); } return this.deployedNode; } return this.rootNode; } computeProjectDisplayName(target, projectOrigin) { const runtimeModel = target.model(SDK.RuntimeModel.RuntimeModel); const executionContexts = runtimeModel ? runtimeModel.executionContexts() : []; for (const context of executionContexts) { if (context.name && context.origin && projectOrigin.startsWith(context.origin)) { return context.name; } } if (!projectOrigin) { return i18nString(UIStrings.noDomain); } const parsedURL = new Common.ParsedURL.ParsedURL(projectOrigin); const prettyURL = parsedURL.isValid ? parsedURL.host + (parsedURL.port ? (':' + parsedURL.port) : '') : ''; return (prettyURL || projectOrigin); } revealUISourceCode(uiSourceCode, select) { const nodes = this.uiSourceCodeNodes.get(uiSourceCode); if (nodes.size === 0) { return null; } const node = nodes.values().next().value; if (!node) { return null; } if (this.scriptsTree.selectedTreeElement) { // If the tree outline is being marked as "being edited" (i.e. we're renaming a file // or chosing the name for a new snippet), we shall not proceed with revealing here, // as that will steal focus from the input widget and thus cancel editing. The // test/e2e/snippets/breakpoint_test.ts exercises this. if (UI.UIUtils.isBeingEdited(this.scriptsTree.selectedTreeElement.treeOutline?.element)) { return null; } this.scriptsTree.selectedTreeElement.deselect(); } this.lastSelectedUISourceCode = uiSourceCode; // TODO(dgozman): figure out revealing multiple. node.reveal(select); return node; } sourceSelected(uiSourceCode, focusSource) { this.lastSelectedUISourceCode = uiSourceCode; void Common.Revealer.reveal(uiSourceCode, !focusSource); } #isUISourceCodeOrAnyAncestorSelected(node) { const selectedTreeElement = this.scriptsTree.selectedTreeElement; const selectedNode = selectedTreeElement && selectedTreeElement.node; let currentNode = node; while (currentNode) { if (currentNode === selectedNode) { return true; } currentNode = currentNode.parent; if (!(node instanceof NavigatorGroupTreeNode || node instanceof NavigatorFolderTreeElement)) { break; } } return false; } removeUISourceCodes(uiSourceCodes) { const nodesWithSelectionOnPath = []; // First we remove source codes without any selection on their path to root, and only then // the ones with selection. This to avoid layout work associated with moving the selection // around (crbug.com/1409025). for (const uiSourceCode of uiSourceCodes) { const nodes = this.uiSourceCodeNodes.get(uiSourceCode); for (const node of nodes) { if (this.#isUISourceCodeOrAnyAncestorSelected(node)) { nodesWithSelectionOnPath.push(node); } else { this.removeUISourceCodeNode(node); } } } nodesWithSelectionOnPath.forEach(this.removeUISourceCodeNode.bind(this)); } removeUISourceCodeNode(node) { const uiSourceCode = node.uiSourceCode(); this.uiSourceCodeNodes.delete(uiSourceCode, node); const project = uiSourceCode.project(); const target = Bindings.NetworkProject.NetworkProject.targetForUISourceCode(uiSourceCode); const frame = node.frame(); let parentNode = node.parent; if (!parentNode) { return; } parentNode.removeChild(node); let currentNode = parentNode; while (currentNode) { parentNode = currentNode.parent; if (!parentNode) { break; } if ((parentNode === this.rootNode || parentNode === this.deployedNode) && project.type() === Workspace.Workspace.projectTypes.FileSystem) { break; } if (!(currentNode instanceof NavigatorGroupTreeNode || currentNode instanceof NavigatorFolderTreeNode)) { break; } if (!currentNode.isEmpty()) { currentNode.updateTitleBubbleUp(); break; } if (currentNode.type === Types.Frame) { this.discardFrame(frame, Boolean(this.groupByAuthored) && uiSourceCode.contentType().isFromSourceMap()); break; } const folderId = this.folderNodeId(project, target, frame, uiSourceCode.origin(), uiSourceCode.contentType().isFromSourceMap(), currentNode instanceof NavigatorFolderTreeNode && currentNode.folderPath || Platform.DevToolsPath.EmptyEncodedPathString); this.subfolderNodes.delete(folderId); parentNode.removeChild(currentNode); if (currentNode === this.authoredNode) { this.authoredNode = undefined; } else if (currentNode === this.deployedNode) { this.deployedNode = undefined; } currentNode = parentNode; } } reset(tearDownOnly) { for (const node of this.uiSourceCodeNodes.valuesArray()) { node.dispose(); } this.scriptsTree.removeChildren(); this.scriptsTree.setFocusable(false); this.uiSourceCodeNodes.clear(); this.subfolderNodes.clear(); this.frameNodes.clear(); this.rootNode.reset(); this.authoredNode = undefined; this.deployedNode = undefined; if (!tearDownOnly) { // Reset the workspace to repopulate filesystem folders. this.resetWorkspace(Workspace.Workspace.WorkspaceImpl.instance()); } } handleContextMenu(_event) { } async renameShortcut() { const selectedTreeElement = this.scriptsTree.selectedTreeElement; const node = selectedTreeElement && selectedTreeElement.node; if (!node || !node.uiSourceCode() || !node.uiSourceCode().canRename()) { return false; } this.rename(node, false); return true; } handleContextMenuCreate(project, path, uiSourceCode) { if (uiSourceCode) { const relativePath = Persistence.FileSystemWorkspaceBinding.FileSystemWorkspaceBinding.relativePath(uiSourceCode); relativePath.pop(); path = Common.ParsedURL.ParsedURL.join(relativePath, '/'); } void this.create(project, path, uiSourceCode); } handleContextMenuRename(node) { this.rename(node, false); } async handleContextMenuExclude(project, path) { const shouldExclude = await UI.UIUtils.ConfirmDialog.show(i18nString(UIStrings.areYouSureYouWantToExcludeThis)); if (shouldExclude) { UI.UIUtils.startBatchUpdate(); project.excludeFolder(Persistence.FileSystemWorkspaceBinding.FileSystemWorkspaceBinding.completeURL(project, path)); UI.UIUtils.endBatchUpdate(); } } async handleContextMenuDelete(uiSourceCode) { const shouldDelete = await UI.UIUtils.ConfirmDialog.show(i18nString(UIStrings.areYouSureYouWantToDeleteThis)); if (shouldDelete) { uiSourceCode.project().deleteFile(uiSourceCode); } } handleFileContextMenu(event, node) { const uiSourceCode = node.uiSourceCode(); const contextMenu = new UI.ContextMenu.ContextMenu(event); contextMenu.appendApplicableItems(uiSourceCode); const project = uiSourceCode.project(); if (project.type() === Workspace.Workspace.projectTypes.FileSystem) { contextMenu.editSection().appendItem(i18nString(UIStrings.rename), this.handleContextMenuRename.bind(this, node)); contextMenu.editSection().appendItem(i18nString(UIStrings.makeACopy), this.handleContextMenuCreate.bind(this, project, Platform.DevToolsPath.EmptyEncodedPathString, uiSourceCode)); contextMenu.editSection().appendItem(i18nString(UIStrings.delete), this.handleContextMenuDelete.bind(this, uiSourceCode)); } void contextMenu.show(); } async handleDeleteOverrides(node) { const shouldRemove = await UI.UIUtils.ConfirmDialog.show(i18nString(UIStrings.areYouSureYouWantToDeleteAll)); if (shouldRemove) { this.handleDeleteOverridesHelper(node); } } handleDeleteOverridesHelper(node) { node.children().forEach(child => { this.handleDeleteOverridesHelper(child); }); if (node instanceof NavigatorUISourceCodeTreeNode) { // Only delete confirmed overrides and not just any file that happens to be in the folder. const binding = Persistence.Persistence.PersistenceImpl.instance().binding(node.uiSourceCode()); if (binding) { node.uiSourceCode().project().deleteFile(node.uiSourceCode()); } } } handleFolderContextMenu(event, node) { const path = node.folderPath || Platform.DevToolsPath.EmptyEncodedPathString; const project = node.project || null; const contextMenu = new UI.ContextMenu.ContextMenu(event); NavigatorView.appendSearchItem(contextMenu, path); if (!project) { return; } if (project.type() === Workspace.Workspace.projectTypes.FileSystem) { const folderPath = Common.ParsedURL.ParsedURL.urlToRawPathString(Persistence.FileSystemWorkspaceBinding.FileSystemWorkspaceBinding.completeURL(project, path), Host.Platform.isWin()); contextMenu.revealSection().appendItem(i18nString(UIStrings.openFolder), () => Host.InspectorFrontendHost.InspectorFrontendHostInstance.showItemInFolder(folderPath)); if (project.canCreateFile()) { contextMenu.defaultSection().appendItem(i18nString(UIStrings.newFile), () => { this.handleContextMenuCreate(project, path, undefined); }); } } else if (node.origin && node.folderPath) { const url = Common.ParsedURL.ParsedURL.concatenate(node.origin, '/', node.folderPath); const options = { isContentScript: node.recursiveProperties.exclusivelyContentScripts || false, isKnownThirdParty: node.recursiveProperties.exclusivelyThirdParty || false, }; for (const { text, callback } of Bindings.IgnoreListManager.IgnoreListManager.instance() .getIgnoreListFolderContextMenuItems(url, options)) { contextMenu.defaultSection().appendItem(text, callback); } } if (project.canExcludeFolder(path)) { contextMenu.defaultSection().appendItem(i18nString(UIStrings.excludeFolder), this.handleContextMenuExclude.bind(this, project, path)); } if (project.type() === Workspace.Workspace.projectTypes.FileSystem) { const isFileOverrides = project.fileSystem().type() === 'overrides'; if (!isFileOverrides) { if (node instanceof NavigatorGroupTreeNode) { contextMenu.defaultSection().appendItem(i18nString(UIStrings.removeFolderFromWorkspace), async () => { const warningMessage = `${i18nString(UIStrings.areYouSureYouWantToRemoveThis, { PH1: node.title, })}\n${i18nString(UIStrings.workspaceStopSyncing)}`; const shouldRemove = await UI.UIUtils.ConfirmDialog.show(warningMessage, undefined, { okButtonLabel: i18nString(UIStrings.remove), }); if (shouldRemove) { project.remove(); } }); } } else { contextMenu.defaultSection().appendItem(i18nString(UIStrings.deleteAllOverrides), this.handleDeleteOverrides.bind(this, node)); } } void contextMenu.show(); } rename(node, creatingNewUISourceCode) { const uiSourceCode = node.uiSourceCode(); node.rename(callback.bind(this)); function callback(committed) { if (!creatingNewUISourceCode) { return; } if (!committed) { uiSourceCode.remove(); } else if (node.treeElement && node.treeElement.listItemElement.hasFocus()) { this.sourceSelected(uiSourceCode, true); } } } async create(project, path, uiSourceCodeToCopy) { let content = ''; if (uiSourceCodeToCopy) { content = (await uiSourceCodeToCopy.requestContent()).content || ''; } const uiSourceCode = await project.createFile(path, null, content); if (!uiSourceCode) { return; } this.sourceSelected(uiSourceCode, false); const node = this.revealUISourceCode(uiSourceCode, true); if (node) { this.rename(node, true); } } groupingChanged() { this.reset(true); this.initGrouping(); // Reset the workspace to repopulate filesystem folders. this.resetWorkspace(Workspace.Workspace.WorkspaceImpl.instance()); this.workspaceInternal.uiSourceCodes().forEach(this.addUISourceCode.bind(this)); } ignoreListChanged() { if (Root.Runtime.experiments.isEnabled(Root.Runtime.ExperimentName.JUST_MY_CODE)) { this.groupingChanged(); } else { this.rootNode.updateTitleRecursive(); } } initGrouping() { this.groupByFrame = true; this.groupByDomain = this.navigatorGroupByFolderSetting.get(); this.groupByFolder = this.groupByDomain; if (this.navigatorGroupByAuthoredExperiment) { this.groupByAuthored = Root.Runtime.experiments.isEnabled(this.navigatorGroupByAuthoredExperiment); } else { this.groupByAuthored = false; } } resetForTest() { this.reset(); this.workspaceInternal.uiSourceCodes().forEach(this.addUISourceCode.bind(this)); } discardFrame(frame, isAuthored) { if (isAuthored) { return; } const node = this.frameNodes.get(frame); if (!node) { return; } if (node.parent) { node.parent.removeChild(node); } this.frameNodes.delete(frame); for (const child of frame.childFrames) { this.discardFrame(child, isAuthored); } } targetAdded(_target) { } targetRemoved(target) { const rootOrDeployed = this.rootOrDeployedNode(); const targetNode = rootOrDeployed.child('target:' + target.id()); if (targetNode) { rootOrDeployed.removeChild(targetNode); } } targetNameChanged(event) { const target = event.data; const targetNode = this.rootOrDeployedNode().child('target:' + target.id()); if (targetNode) { targetNode.setTitle(target.name()); } } wasShown() { super.wasShown(); this.scriptsTree.registerCSSFiles([navigatorTreeStyles]); this.registerCSSFiles([navigatorViewStyles]); } } const boostOrderForNode = new WeakSet(); export class NavigatorFolderTreeElement extends UI.TreeOutline.TreeElement { nodeType; navigatorView; hoverCallback; node; hovered; isIgnoreListed; isFromSourceMap; constructor(navigatorView, type, title, hoverCallback) { super('', true); this.listItemElement.classList.add('navigator-' + type + '-tree-item', 'navigator-folder-tree-item'); UI.ARIAUtils.setLabel(this.listItemElement, `${title}, ${type}`); this.nodeType = type; this.title = title; this.tooltip = title; this.navigatorView = navigatorView; this.hoverCallback = hoverCallback; this.isFromSourceMap = false; let iconType = 'folder'; if (type === Types.Domain) { iconType = 'cloud'; } else if (type === Types.Frame) { iconType = 'frame'; } else if (type === Types.Worker) { iconType = 'gears'; } else if (type === Types.Authored) { iconType = 'code'; } else if (type === Types.Deployed) { iconType = 'deployed'; } const icon = new IconButton.Icon.Icon(); const iconPath = new URL(`../../Images/${iconType}.svg`, import.meta.url).toString(); icon.data = { iconPath: iconPath, color: 'var(--override-folder-tree-item-color)', width: '20px', height: '20px' }; this.setLeadingIcons([icon]); } async onpopulate() { this.node.populate(); } onattach() { this.collapse(); this.node.onattach(); this.listItemElement.addEventListener('contextmenu', this.handleContextMenuEvent.bind(this), false); this.listItemElement.addEventListener('mousemove', this.mouseMove.bind(this), false); this.listItemElement.addEventListener('mouseleave', this.mouseLeave.bind(this), false); } setIgnoreListed(isIgnoreListed) { if (this.isIgnoreListed !== isIgnoreListed) { this.isIgnoreListed = isIgnoreListed; this.listItemElement.classList.toggle('is-ignore-listed', isIgnoreListed); this.updateTooltip(); } } setFromSourceMap(isFromSourceMap) { this.isFromSourceMap = isFromSourceMap; this.listItemElement.classList.toggle('is-from-source-map', isFromSourceMap); } setNode(node) { this.node = node; this.updateTooltip(); UI.ARIAUtils.setLabel(this.listItemElement, `${this.title}, ${this.nodeType}`); } updateTooltip() { if (this.node.tooltip) { this.tooltip = this.node.tooltip; } else { const paths = []; let currentNode = this.node; while (currentNode && !currentNode.isRoot() && currentNode.type === this.node.type) { paths.push(currentNode.title); currentNode = currentNode.parent; } paths.reverse(); let tooltip = paths.join('/'); if (this.isIgnoreListed) { tooltip = i18nString(UIStrings.sIgnoreListed, { PH1: tooltip }); } this.tooltip = tooltip; } } handleContextMenuEvent(event) { if (!this.node) { return; } this.select(); this.navigatorView.handleFolderContextMenu(event, this.node); } mouseMove(_event) { if (this.hovered || !this.hoverCallback) { return; } this.hovered = true; this.hoverCallback(true); } mouseLeave(_event) { if (!this.hoverCallback) { return; } this.hovered = false; this.hoverCallback(false); } } export class NavigatorSourceTreeElement extends UI.TreeOutline.TreeElement { nodeType; node; navigatorView; uiSourceCodeInternal; constructor(navigatorView, uiSourceCode, title, node) { super('', false); this.nodeType = Types.File; this.node = node; this.title = title; this.listItemElement.classList.add('navigator-' + uiSourceCode.contentType().name() + '-tree-item', 'navigator-file-tree-item'); this.tooltip = uiSourceCode.url(); UI.ARIAUtils.setLabel(this.listItemElement, `${uiSourceCode.name()}, ${this.nodeType}`); Common.EventTarget.fireEvent('source-tree-file-added', uiSourceCode.fullDisplayName()); this.navigatorView = navigatorView; this.uiSourceCodeInternal = uiSourceCode; this.updateIcon(); } updateIcon() { const binding = Persistence.Persistence.PersistenceImpl.instance().binding(this.uiSourceCodeInternal); let iconType = 'document'; let iconStyles = []; if (binding) { if (Snippets.ScriptSnippetFileSystem.isSnippetsUISourceCode(binding.fileSystem)) { iconType = 'snippet'; } const badgeIsPurple = Persistence.NetworkPersistence