chrome-devtools-frontend
Version:
Chrome DevTools UI
245 lines (215 loc) • 9.55 kB
text/typescript
/*
* Copyright (C) 2013 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 type * as Common from '../../core/common/common.js';
import * as i18n from '../../core/i18n/i18n.js';
import type * as Platform from '../../core/platform/platform.js';
import type * as TextUtils from '../text_utils/text_utils.js';
import * as Workspace from '../workspace/workspace.js';
const UIStrings = {
/**
* @description Error message that is displayed in the Sources panel when can't be loaded.
*/
unknownErrorLoadingFile: 'Unknown error loading file',
};
const str_ = i18n.i18n.registerUIStrings('models/bindings/ContentProviderBasedProject.ts', UIStrings);
const i18nString = i18n.i18n.getLocalizedString.bind(undefined, str_);
interface UISourceCodeData {
mimeType: string;
metadata: Workspace.UISourceCode.UISourceCodeMetadata|null;
}
export class ContentProviderBasedProject extends Workspace.Workspace.ProjectStore {
readonly #contentProviders: Map<string, TextUtils.ContentProvider.ContentProvider>;
readonly #isServiceProjectInternal: boolean;
readonly #uiSourceCodeToData: WeakMap<Workspace.UISourceCode.UISourceCode, UISourceCodeData>;
constructor(
workspace: Workspace.Workspace.WorkspaceImpl, id: string, type: Workspace.Workspace.projectTypes,
displayName: string, isServiceProject: boolean) {
super(workspace, id, type, displayName);
this.#contentProviders = new Map();
this.#isServiceProjectInternal = isServiceProject;
this.#uiSourceCodeToData = new WeakMap();
workspace.addProject(this);
}
async requestFileContent(uiSourceCode: Workspace.UISourceCode.UISourceCode):
Promise<TextUtils.ContentProvider.DeferredContent> {
const contentProvider =
(this.#contentProviders.get(uiSourceCode.url()) as TextUtils.ContentProvider.ContentProvider);
try {
const content = await contentProvider.requestContent();
const wasmDisassemblyInfo = 'wasmDisassemblyInfo' in content ? content.wasmDisassemblyInfo : undefined;
return {
content: content.content,
wasmDisassemblyInfo,
isEncoded: content.isEncoded,
error: 'error' in content && content.error || '',
};
} catch (err) {
// TODO(rob.paveza): CRBug 1013683 - Consider propagating exceptions full-stack
return {
content: null,
isEncoded: false,
error: err ? String(err) : i18nString(UIStrings.unknownErrorLoadingFile),
};
}
}
isServiceProject(): boolean {
return this.#isServiceProjectInternal;
}
async requestMetadata(uiSourceCode: Workspace.UISourceCode.UISourceCode):
Promise<Workspace.UISourceCode.UISourceCodeMetadata|null> {
const {metadata} = (this.#uiSourceCodeToData.get(uiSourceCode) as UISourceCodeData);
return metadata;
}
canSetFileContent(): boolean {
return false;
}
async setFileContent(_uiSourceCode: Workspace.UISourceCode.UISourceCode, _newContent: string, _isBase64: boolean):
Promise<void> {
}
fullDisplayName(uiSourceCode: Workspace.UISourceCode.UISourceCode): string {
let parentPath = uiSourceCode.parentURL().replace(/^(?:https?|file)\:\/\//, '');
try {
parentPath = decodeURI(parentPath);
} catch (e) {
}
return parentPath + '/' + uiSourceCode.displayName(true);
}
mimeType(uiSourceCode: Workspace.UISourceCode.UISourceCode): string {
const {mimeType} = (this.#uiSourceCodeToData.get(uiSourceCode) as UISourceCodeData);
return mimeType;
}
canRename(): boolean {
return false;
}
rename(
uiSourceCode: Workspace.UISourceCode.UISourceCode, newName: Platform.DevToolsPath.RawPathString,
callback:
(arg0: boolean, arg1?: string|undefined, arg2?: Platform.DevToolsPath.UrlString|undefined,
arg3?: Common.ResourceType.ResourceType|undefined) => void): void {
const path = uiSourceCode.url();
this.performRename(path, newName, innerCallback.bind(this));
function innerCallback(this: ContentProviderBasedProject, success: boolean, newName?: string): void {
if (success && newName) {
const copyOfPath = path.split('/');
copyOfPath[copyOfPath.length - 1] = newName;
const newPath = copyOfPath.join('/');
const contentProvider = (this.#contentProviders.get(path) as TextUtils.ContentProvider.ContentProvider);
this.#contentProviders.set(newPath, contentProvider);
this.#contentProviders.delete(path);
this.renameUISourceCode(uiSourceCode, newName);
}
callback(success, newName);
}
}
excludeFolder(_path: Platform.DevToolsPath.UrlString): void {
}
canExcludeFolder(_path: Platform.DevToolsPath.EncodedPathString): boolean {
return false;
}
async createFile(
_path: Platform.DevToolsPath.EncodedPathString, _name: string|null, _content: string,
_isBase64?: boolean): Promise<Workspace.UISourceCode.UISourceCode|null> {
return null;
}
canCreateFile(): boolean {
return false;
}
deleteFile(_uiSourceCode: Workspace.UISourceCode.UISourceCode): void {
}
remove(): void {
}
performRename(
path: Platform.DevToolsPath.UrlString, newName: string,
callback: (arg0: boolean, arg1?: string|undefined) => void): void {
callback(false);
}
searchInFileContent(
uiSourceCode: Workspace.UISourceCode.UISourceCode, query: string, caseSensitive: boolean,
isRegex: boolean): Promise<TextUtils.ContentProvider.SearchMatch[]> {
const contentProvider =
(this.#contentProviders.get(uiSourceCode.url()) as TextUtils.ContentProvider.ContentProvider);
return contentProvider.searchInContent(query, caseSensitive, isRegex);
}
async findFilesMatchingSearchRequest(
searchConfig: Workspace.Workspace.ProjectSearchConfig, filesMatchingFileQuery: Platform.DevToolsPath.UrlString[],
progress: Common.Progress.Progress): Promise<string[]> {
const result: string[] = [];
progress.setTotalWork(filesMatchingFileQuery.length);
await Promise.all(filesMatchingFileQuery.map(searchInContent.bind(this)));
progress.done();
return result;
async function searchInContent(this: ContentProviderBasedProject, path: string): Promise<void> {
const contentProvider = (this.#contentProviders.get(path) as TextUtils.ContentProvider.ContentProvider);
let allMatchesFound = true;
for (const query of searchConfig.queries().slice()) {
const searchMatches =
await contentProvider.searchInContent(query, !searchConfig.ignoreCase(), searchConfig.isRegex());
if (!searchMatches.length) {
allMatchesFound = false;
break;
}
}
if (allMatchesFound) {
result.push(path);
}
progress.incrementWorked(1);
}
}
indexContent(progress: Common.Progress.Progress): void {
queueMicrotask(progress.done.bind(progress));
}
addUISourceCodeWithProvider(
uiSourceCode: Workspace.UISourceCode.UISourceCode, contentProvider: TextUtils.ContentProvider.ContentProvider,
metadata: Workspace.UISourceCode.UISourceCodeMetadata|null, mimeType: string): void {
this.#contentProviders.set(uiSourceCode.url(), contentProvider);
this.#uiSourceCodeToData.set(uiSourceCode, {mimeType, metadata});
this.addUISourceCode(uiSourceCode);
}
addContentProvider(
url: Platform.DevToolsPath.UrlString, contentProvider: TextUtils.ContentProvider.ContentProvider,
mimeType: string): Workspace.UISourceCode.UISourceCode {
const uiSourceCode = this.createUISourceCode(url, contentProvider.contentType());
this.addUISourceCodeWithProvider(uiSourceCode, contentProvider, null, mimeType);
return uiSourceCode;
}
removeFile(path: Platform.DevToolsPath.UrlString): void {
this.#contentProviders.delete(path);
this.removeUISourceCode(path);
}
reset(): void {
this.#contentProviders.clear();
this.removeProject();
this.workspace().addProject(this);
}
dispose(): void {
this.#contentProviders.clear();
this.removeProject();
}
}