UNPKG

chrome-devtools-frontend

Version:
267 lines (246 loc) 9.65 kB
/* * Copyright (C) 2007, 2008 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. * 3. Neither the name of Apple Computer, Inc. ("Apple") 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 APPLE 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 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. */ import * as Common from '../common/common.js'; import * as Host from '../host/host.js'; import * as i18n from '../i18n/i18n.js'; import * as Platform from '../platform/platform.js'; import * as TextUtils from '../text_utils/text_utils.js'; import * as UI from '../ui/ui.js'; import * as Workspace from '../workspace/workspace.js'; export const UIStrings = { /** *@description Text in Image View of the Sources panel */ image: 'Image', /** *@description Text that appears when user drag and drop something (for example, a file) in Image View of the Sources panel */ dropImageFileHere: 'Drop image file here', /** *@description Text to indicate the source of an image *@example {example.com} PH1 */ imageFromS: 'Image from {PH1}', /** *@description Text in Image View of the Sources panel *@example {2} PH1 *@example {2} PH2 */ dD: '{PH1} × {PH2}', /** *@description A context menu item in the Image View of the Sources panel */ copyImageUrl: 'Copy image URL', /** *@description A context menu item in the Image View of the Sources panel */ copyImageAsDataUri: 'Copy image as data URI', /** *@description A context menu item in the Image View of the Sources panel */ openImageInNewTab: 'Open image in new tab', /** *@description A context menu item in the Image Preview */ saveImageAs: 'Save image as...', /** *@description The default file name when downloading a file */ download: 'download', }; const str_ = i18n.i18n.registerUIStrings('source_frame/ImageView.js', UIStrings); const i18nString = i18n.i18n.getLocalizedString.bind(undefined, str_); export class ImageView extends UI.View.SimpleView { /** * @param {string} mimeType * @param {!TextUtils.ContentProvider.ContentProvider} contentProvider */ constructor(mimeType, contentProvider) { super(i18nString(UIStrings.image)); this.registerRequiredCSS('source_frame/imageView.css', {enableLegacyPatching: false}); this.element.tabIndex = -1; this.element.classList.add('image-view'); this._url = contentProvider.contentURL(); this._parsedURL = new Common.ParsedURL.ParsedURL(this._url); this._mimeType = mimeType; this._contentProvider = contentProvider; this._uiSourceCode = contentProvider instanceof Workspace.UISourceCode.UISourceCode ? /** @type {!Workspace.UISourceCode.UISourceCode} */ (contentProvider) : null; if (this._uiSourceCode) { this._uiSourceCode.addEventListener( Workspace.UISourceCode.Events.WorkingCopyCommitted, this._workingCopyCommitted, this); new UI.DropTarget.DropTarget( this.element, [UI.DropTarget.Type.ImageFile, UI.DropTarget.Type.URI], i18nString(UIStrings.dropImageFileHere), this._handleDrop.bind(this)); } this._sizeLabel = new UI.Toolbar.ToolbarText(); this._dimensionsLabel = new UI.Toolbar.ToolbarText(); this._mimeTypeLabel = new UI.Toolbar.ToolbarText(mimeType); this._container = this.element.createChild('div', 'image'); /** @type {!HTMLImageElement} */ this._imagePreviewElement = /** @type {!HTMLImageElement} */ (this._container.createChild('img', 'resource-image-view')); this._imagePreviewElement.addEventListener('contextmenu', this._contextMenu.bind(this), true); } /** * @override * @return {!Promise<!Array<!UI.Toolbar.ToolbarItem>>} */ async toolbarItems() { await this._updateContentIfNeeded(); return [ this._sizeLabel, new UI.Toolbar.ToolbarSeparator(), this._dimensionsLabel, new UI.Toolbar.ToolbarSeparator(), this._mimeTypeLabel ]; } /** * @override */ wasShown() { this._updateContentIfNeeded(); } /** * @override */ disposeView() { if (this._uiSourceCode) { this._uiSourceCode.removeEventListener( Workspace.UISourceCode.Events.WorkingCopyCommitted, this._workingCopyCommitted, this); } } _workingCopyCommitted() { this._updateContentIfNeeded(); } async _updateContentIfNeeded() { const {content} = await this._contentProvider.requestContent(); if (this._cachedContent === content) { return; } const contentEncoded = await this._contentProvider.contentEncoded(); /** @type {?string} */ this._cachedContent = content; const imageSrc = TextUtils.ContentProvider.contentAsDataURL(content, this._mimeType, contentEncoded) || this._url; const loadPromise = new Promise(x => { this._imagePreviewElement.onload = x; }); this._imagePreviewElement.src = imageSrc; this._imagePreviewElement.alt = i18nString(UIStrings.imageFromS, {PH1: this._url}); const size = content && !contentEncoded ? content.length : base64ToSize(content); this._sizeLabel.setText(Platform.NumberUtilities.bytesToString(size)); await loadPromise; this._dimensionsLabel.setText(i18nString( UIStrings.dD, {PH1: this._imagePreviewElement.naturalWidth, PH2: this._imagePreviewElement.naturalHeight})); } /** * @param {!Event} event */ _contextMenu(event) { const contextMenu = new UI.ContextMenu.ContextMenu(event); const parsedSrc = new Common.ParsedURL.ParsedURL(this._imagePreviewElement.src); if (!this._parsedURL.isDataURL()) { contextMenu.clipboardSection().appendItem(i18nString(UIStrings.copyImageUrl), this._copyImageURL.bind(this)); } if (parsedSrc.isDataURL()) { contextMenu.clipboardSection().appendItem( i18nString(UIStrings.copyImageAsDataUri), this._copyImageAsDataURL.bind(this)); } contextMenu.clipboardSection().appendItem(i18nString(UIStrings.openImageInNewTab), this._openInNewTab.bind(this)); contextMenu.clipboardSection().appendItem(i18nString(UIStrings.saveImageAs), async () => { await this._saveImage(); }); contextMenu.show(); } _copyImageAsDataURL() { Host.InspectorFrontendHost.InspectorFrontendHostInstance.copyText(this._imagePreviewElement.src); } _copyImageURL() { Host.InspectorFrontendHost.InspectorFrontendHostInstance.copyText(this._url); } async _saveImage() { const contentEncoded = await this._contentProvider.contentEncoded(); if (!this._cachedContent) { return; } const cachedContent = this._cachedContent; const imageDataURL = TextUtils.ContentProvider.contentAsDataURL(cachedContent, this._mimeType, contentEncoded, '', false); if (!imageDataURL) { return; } const link = document.createElement('a'); link.href = imageDataURL; // If it is a Base64 image, set a default file name. // When chrome saves a file, the file name characters that are not supported // by the OS will be replaced automatically. For example, in the Mac, // `:` it will be replaced with `_`. link.download = this._parsedURL.isDataURL() ? i18nString(UIStrings.download) : decodeURIComponent(this._parsedURL.displayName); link.click(); link.remove(); } _openInNewTab() { Host.InspectorFrontendHost.InspectorFrontendHostInstance.openInNewTab(this._url); } /** * @param {!DataTransfer} dataTransfer */ async _handleDrop(dataTransfer) { const items = dataTransfer.items; if (!items.length || items[0].kind !== 'file') { return; } const entry = items[0].webkitGetAsEntry(); const encoded = !entry.name.endsWith('.svg'); /** * @param {!Blob} file */ const fileCallback = file => { const reader = new FileReader(); reader.onloadend = () => { let result; try { result = /** @type {?string} */ (reader.result); } catch (e) { result = null; console.error('Can\'t read file: ' + e); } if (typeof result !== 'string' || !this._uiSourceCode) { return; } this._uiSourceCode.setContent(encoded ? btoa(result) : result, encoded); }; if (encoded) { reader.readAsBinaryString(file); } else { reader.readAsText(file); } }; entry.file(fileCallback); } }