blob: ee5b0457827bb6b6bf63915a7ef4b72476474f0a [file] [log] [blame]
// Copyright 2013 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
import {FileType} from 'chrome://file-manager/common/js/file_type.js';
import {assert, assertInstanceof} from 'chrome://resources/ash/common/assert.js';
import {ImageCache} from './cache.js';
import {ImageLoaderUtil} from './image_loader_util.js';
import {ImageOrientation} from './image_orientation.js';
import {LoadImageRequest, LoadImageResponse, LoadImageResponseStatus} from './load_image_request.js';
import {PiexLoader} from './piex_loader.js';
/**
* Creates and starts downloading and then resizing of the image. Finally,
* returns the image using the callback.
*
* @param {string} id Request ID.
* @param {ImageCache} cache Cache object.
* @param {!LoadImageRequest} request Request message as a hash array.
* @param {function(!LoadImageResponse)} callback Response handler.
* @constructor
*/
export function ImageRequestTask(id, cache, request, callback) {
/**
* Global ID (concatenated client ID and client request ID).
* @type {string}
* @private
*/
this.id_ = id;
/**
* @type {ImageCache}
* @private
*/
this.cache_ = cache;
/**
* @type {!LoadImageRequest}
* @private
*/
this.request_ = request;
/**
* @type {function(!LoadImageResponse)}
* @private
*/
this.sendResponse_ = callback;
/**
* Temporary image used to download images.
* @type {Image}
* @private
*/
this.image_ = new Image();
/**
* MIME type of the fetched image.
* @type {?string}
* @private
*/
this.contentType_ = null;
/**
* IFD data of the fetched image. Only RAW images provide a non-null
* ifd at this time. Drive images might provide an ifd in future.
* @type {?string}
* @private
*/
this.ifd_ = null;
/**
* Used to download remote images using http:// or https:// protocols.
* @type {XMLHttpRequest}
* @private
*/
this.xhr_ = null;
/**
* Temporary canvas used to resize and compress the image.
* @type {HTMLCanvasElement}
* @private
*/
this.canvas_ =
/** @type {HTMLCanvasElement} */ (document.createElement('canvas'));
/**
* @type {CanvasRenderingContext2D}
* @private
*/
this.context_ =
/** @type {CanvasRenderingContext2D} */ (this.canvas_.getContext('2d'));
/**
* @type {ImageOrientation|null}
*/
this.renderOrientation_ = null;
/**
* Callback to be called once downloading is finished.
* @type {?function()}
* @private
*/
this.downloadCallback_ = null;
/**
* @type {boolean}
* @private
*/
this.aborted_ = false;
}
/**
* Seeks offset to generate video thumbnail.
* TODO(ryoh):
* What is the best position for the thumbnail?
* The first frame seems not good -- sometimes it is a black frame.
* @const
* @type {number}
*/
ImageRequestTask.VIDEO_THUMBNAIL_POSITION = 3; // [sec]
/**
* The maximum milliseconds to load video. If loading video exceeds the limit,
* we give up generating video thumbnail and free the consumed memory.
* @const
* @type {number}
*/
ImageRequestTask.MAX_MILLISECONDS_TO_LOAD_VIDEO = 3000;
/**
* The default size (width and height) of a square thumbnail. The value is set
* to match the behavior of drivefs thumbnail generation.
* See chromeos/ash/components/drivefs/mojom/drivefs.mojom
* @const
* @type {number}
*/
ImageRequestTask.DEFAULT_THUMBNAIL_SQUARE_SIZE = 360;
/**
* The default width of a non-square thumbnail. The value is set to match the
* behavior of drivefs thumbnail generation.
* See chromeos/ash/components/drivefs/mojom/drivefs.mojom
* @const
* @type {number}
*/
ImageRequestTask.DEFAULT_THUMBNAIL_WIDTH = 500;
/**
* The default height of a non-square thumbnail. The value is set to match the
* behavior of drivefs thumbnail generation.
* See chromeos/ash/components/drivefs/mojom/drivefs.mojom
* @const
* @type {number}
*/
ImageRequestTask.DEFAULT_THUMBNAIL_HEIGHT = 500;
/**
* A map which is used to estimate content type from extension.
* @enum {string}
*/
ImageRequestTask.ExtensionContentTypeMap = {
gif: 'image/gif',
png: 'image/png',
svg: 'image/svg',
bmp: 'image/bmp',
jpg: 'image/jpeg',
jpeg: 'image/jpeg',
};
/**
* Extracts MIME type of a data URL.
* @param {string|undefined} dataUrl Data URL.
* @return {?string} MIME type string, or null if the URL is invalid.
*/
ImageRequestTask.getDataUrlMimeType = function(dataUrl) {
const dataUrlMatches = (dataUrl || '').match(/^data:([^,;]*)[,;]/);
return dataUrlMatches ? dataUrlMatches[1] : null;
};
/**
* Returns ID of the request.
* @return {string} Request ID.
*/
ImageRequestTask.prototype.getId = function() {
return this.id_;
};
/**
* Returns the client's task ID for the request.
* @return {number}
*/
ImageRequestTask.prototype.getClientTaskId = function() {
// Every incoming request should have been given a taskId.
assert(this.request_.taskId);
return this.request_.taskId;
};
/**
* Returns priority of the request. The higher priority, the faster it will
* be handled. The highest priority is 0. The default one is 2.
*
* @return {number} Priority.
*/
ImageRequestTask.prototype.getPriority = function() {
return (this.request_.priority !== undefined) ? this.request_.priority : 2;
};
/**
* Tries to load the image from cache, if it exists in the cache, and sends
* the response. Fails if the image is not found in the cache.
*
* @param {function()} onSuccess Success callback.
* @param {function()} onFailure Failure callback.
*/
ImageRequestTask.prototype.loadFromCacheAndProcess = function(
onSuccess, onFailure) {
this.loadFromCache_(
function(width, height, ifd, data) { // Found in cache.
this.ifd_ = ifd;
this.sendImageData_(width, height, data);
onSuccess();
}.bind(this),
onFailure); // Not found in cache.
};
/**
* Tries to download the image, resizes and sends the response.
*
* @param {function()} callback Completion callback.
*/
ImageRequestTask.prototype.downloadAndProcess = function(callback) {
if (this.downloadCallback_) {
throw new Error('Downloading already started.');
}
this.downloadCallback_ = callback;
this.downloadThumbnail_(
this.onImageLoad_.bind(this), this.onImageError_.bind(this));
};
/**
* Fetches the image from the persistent cache.
*
* @param {function(number, number, ?string, string)} onSuccess
* Success callback with the image width, height, ?ifd, and data.
* @param {function()} onFailure Failure callback.
* @private
*/
ImageRequestTask.prototype.loadFromCache_ = function(onSuccess, onFailure) {
const cacheKey = LoadImageRequest.cacheKey(this.request_);
if (!cacheKey) {
// Cache key is not provided for the request.
onFailure();
return;
}
if (!this.request_.cache) {
// Cache is disabled for this request; therefore, remove it from cache
// if existed.
this.cache_.removeImage(cacheKey);
onFailure();
return;
}
const timestamp = this.request_.timestamp;
if (!timestamp) {
// Persistent cache is available only when a timestamp is provided.
onFailure();
return;
}
this.cache_.loadImage(cacheKey, timestamp, onSuccess, onFailure);
};
/**
* Saves the image to the persistent cache.
*
* @param {number} width Image width.
* @param {number} height Image height.
* @param {string} data Image data.
* @private
*/
ImageRequestTask.prototype.saveToCache_ = function(width, height, data) {
const timestamp = this.request_.timestamp;
if (!this.request_.cache || !timestamp) {
// Persistent cache is available only when a timestamp is provided.
return;
}
const cacheKey = LoadImageRequest.cacheKey(this.request_);
if (!cacheKey) {
// Cache key is not provided for the request.
return;
}
this.cache_.saveImage(cacheKey, timestamp, width, height, this.ifd_, data);
};
/**
* Gets the target image size for external thumbnails, where supported.
The defaults replicate drivefs thumbnailer behavior.
* @return {{width: !number, height: !number}}
*/
ImageRequestTask.prototype.targetThumbnailSize_ = function() {
const crop = !!this.request_.crop;
const defaultWidth = crop ? ImageRequestTask.DEFAULT_THUMBNAIL_SQUARE_SIZE :
ImageRequestTask.DEFAULT_THUMBNAIL_WIDTH;
const defaultHeight = crop ? ImageRequestTask.DEFAULT_THUMBNAIL_SQUARE_SIZE :
ImageRequestTask.DEFAULT_THUMBNAIL_HEIGHT;
return {
width: this.request_.width || defaultWidth,
height: this.request_.height || defaultHeight,
};
};
/**
* Loads |this.image_| with the |this.request_.url| source or the thumbnail
* image of the source.
*
* @param {function()} onSuccess Success callback.
* @param {function()} onFailure Failure callback.
* @private
*/
ImageRequestTask.prototype.downloadThumbnail_ = function(onSuccess, onFailure) {
// Load methods below set |this.image_.src|. Call revokeObjectURL(src) to
// release resources if the image src was created with createObjectURL().
this.image_.onload = function() {
URL.revokeObjectURL(this.image_.src);
onSuccess();
}.bind(this);
this.image_.onerror = function() {
URL.revokeObjectURL(this.image_.src);
onFailure();
}.bind(this);
// Load dataURL sources directly.
const dataUrlMimeType =
ImageRequestTask.getDataUrlMimeType(this.request_.url);
if (dataUrlMimeType) {
this.image_.src = this.request_.url;
this.contentType_ = dataUrlMimeType;
return;
}
const resolveLocalFileSystemUrl = (url, onResolveSuccess) => {
window.webkitResolveLocalFileSystemURL(url, onResolveSuccess, error => {
console.warn(error);
onFailure();
});
};
const onExternalThumbnail = (dataUrl) => {
if (chrome.runtime.lastError) {
console.warn(chrome.runtime.lastError.message);
onFailure();
} else if (dataUrl) {
this.image_.src = dataUrl;
this.contentType_ = ImageRequestTask.getDataUrlMimeType(dataUrl);
} else {
onFailure();
}
};
// Load Drive source thumbnail.
const drivefsUrlMatches = this.request_.url.match(/^drivefs:(.*)/);
if (drivefsUrlMatches) {
const url = drivefsUrlMatches[1];
const cropToSquare = !!this.request_.crop;
resolveLocalFileSystemUrl(
url,
entry => chrome.fileManagerPrivate.getDriveThumbnail(
entry, cropToSquare, onExternalThumbnail));
return;
}
// Load PDF source thumbnail.
if (this.request_.url.endsWith('.pdf')) {
const {width, height} = this.targetThumbnailSize_();
resolveLocalFileSystemUrl(
this.request_.url,
entry => chrome.fileManagerPrivate.getPdfThumbnail(
entry, width, height, onExternalThumbnail));
return;
}
// Load DocumentsProvider thumbnail, if supported.
const isDocumentsProviderRequest = !!this.request_.url.match(RegExp(
'filesystem:chrome-extension://[a-z]+/external/arc-documents-provider/.*'));
if (isDocumentsProviderRequest) {
const {width, height} = this.targetThumbnailSize_();
resolveLocalFileSystemUrl(this.request_.url, entry => {
chrome.fileManagerPrivate.getArcDocumentsProviderThumbnail(
entry, width, height, onExternalThumbnail);
});
return;
}
const fileType = FileType.getTypeForName(this.request_.url);
// Load RAW image source thumbnail.
if (fileType.type === 'raw') {
PiexLoader.load(this.request_.url, chrome.runtime.reload)
.then(
function(data) {
this.renderOrientation_ =
ImageOrientation.fromExifOrientation(data.orientation);
this.ifd_ = data.ifd;
this.contentType_ = data.mimeType;
const blob = new Blob([data.thumbnail], {type: data.mimeType});
this.image_.src = URL.createObjectURL(blob);
}.bind(this),
function() {
// PiexLoader calls console.warn on errors.
onFailure();
});
return;
}
// Load video source thumbnail.
if (fileType.type === 'video') {
this.createVideoThumbnailUrl_(this.request_.url)
.then(function(url) {
this.image_.src = url;
}.bind(this))
.catch(function(error) {
console.warn('Video thumbnail error: ', error);
onFailure();
});
return;
}
// Load the source directly.
this.load(this.request_.url, (contentType, blob) => {
this.image_.src = blob ? URL.createObjectURL(blob) : '!';
this.contentType_ = contentType || null;
if (this.contentType_ === 'image/jpeg') {
this.renderOrientation_ = ImageOrientation.fromExifOrientation(1);
}
}, onFailure);
};
/**
* Creates a video thumbnail data url from video file.
*
* @param {string} url Video URL.
* @return {!Promise<Blob>} Promise that resolves with the data url of video
* thumbnail.
* @private
*/
ImageRequestTask.prototype.createVideoThumbnailUrl_ = function(url) {
const video =
assertInstanceof(document.createElement('video'), HTMLVideoElement);
return Promise
.race([
new Promise((resolve, reject) => {
video.addEventListener('canplay', resolve);
video.addEventListener('error', reject);
video.currentTime = ImageRequestTask.VIDEO_THUMBNAIL_POSITION;
video.preload = 'auto';
video.src = url;
video.load();
}),
new Promise((resolve) => {
setTimeout(resolve, ImageRequestTask.MAX_MILLISECONDS_TO_LOAD_VIDEO);
}).then(() => {
// If we don't receive 'canplay' event after 3 seconds have passed for
// some reason (e.g. unseekable video), we give up generating
// thumbnail.
video.src =
''; // Make sure to stop loading remaining part of the video.
throw new Error('Seeking video failed.');
}),
])
.then(() => {
const canvas = assertInstanceof(
document.createElement('canvas'), HTMLCanvasElement);
canvas.width = video.videoWidth;
canvas.height = video.videoHeight;
assertInstanceof(canvas.getContext('2d'), CanvasRenderingContext2D)
.drawImage(video, 0, 0);
return canvas.toDataURL();
});
};
/**
* Loads an image.
*
* @param {string} url URL to the resource to be fetched.
* @param {function(string, Blob)} onSuccess Success callback with the content
* type and the fetched data.
* @param {function()} onFailure Failure callback.
*/
ImageRequestTask.prototype.load = function(url, onSuccess, onFailure) {
this.aborted_ = false;
// Do not call any callbacks when aborting.
const onMaybeSuccess =
/** @type {function(string, Blob)} */ (function(contentType, response) {
// When content type is not available, try to estimate it from url.
if (!contentType) {
contentType =
ImageRequestTask
.ExtensionContentTypeMap[this.extractExtension_(url)];
}
if (!this.aborted_) {
onSuccess(contentType, response);
}
}.bind(this));
const onMaybeFailure = /** @type {function(number=)} */ (function(opt_code) {
if (!this.aborted_) {
onFailure();
}
}.bind(this));
// The query parameter is workaround for crbug.com/379678, which forces the
// browser to obtain the latest contents of the image.
const noCacheUrl = url + '?nocache=' + Date.now();
this.xhr_ =
ImageRequestTask.load_(noCacheUrl, onMaybeSuccess, onMaybeFailure);
};
/**
* Extracts extension from url.
* @param {string} url Url.
* @return {string} Extracted extension, e.g. png.
*/
ImageRequestTask.prototype.extractExtension_ = function(url) {
const result = (/\.([a-zA-Z]+)$/i).exec(url);
return result ? result[1] : '';
};
/**
* Fetches data using XmlHttpRequest.
*
* @param {string} url URL to the resource to be fetched.
* @param {function(string, Blob)} onSuccess Success callback with the content
* type and the fetched data.
* @param {function(number=)} onFailure Failure callback with the error code
* if available.
* @return {XMLHttpRequest} XHR instance.
* @private
*/
ImageRequestTask.load_ = function(url, onSuccess, onFailure) {
const xhr = new XMLHttpRequest();
xhr.responseType = 'blob';
xhr.onreadystatechange = function() {
if (xhr.readyState != 4) {
return;
}
if (xhr.status != 200) {
onFailure(xhr.status);
return;
}
const response = /** @type {Blob} */ (xhr.response);
const contentType = xhr.getResponseHeader('Content-Type') || response.type;
onSuccess(contentType, response);
}.bind(this);
// Perform a xhr request.
try {
xhr.open('GET', url, true);
xhr.send();
} catch (e) {
onFailure();
}
return xhr;
};
/**
* Sends the resized image via the callback. If the image has been changed,
* then packs the canvas contents, otherwise sends the raw image data.
*
* @param {boolean} imageChanged Whether the image has been changed.
* @private
*/
ImageRequestTask.prototype.sendImage_ = function(imageChanged) {
let width;
let height;
let data;
if (!imageChanged) {
// The image hasn't been processed, so the raw data can be directly
// forwarded for speed (no need to encode the image again).
width = this.image_.width;
height = this.image_.height;
data = this.image_.src;
} else {
// The image has been resized or rotated, therefore the canvas has to be
// encoded to get the correct compressed image data.
width = this.canvas_.width;
height = this.canvas_.height;
switch (this.contentType_) {
case 'image/gif':
case 'image/png':
case 'image/svg':
case 'image/bmp':
data = this.canvas_.toDataURL('image/png');
break;
case 'image/jpeg':
default:
data = this.canvas_.toDataURL('image/jpeg', 0.9);
break;
}
}
// Send the image data and also save it in the persistent cache.
this.sendImageData_(width, height, data);
this.saveToCache_(width, height, data);
};
/**
* Sends the resized image via the callback.
*
* @param {number} width Image width.
* @param {number} height Image height.
* @param {string} data Image data.
* @private
*/
ImageRequestTask.prototype.sendImageData_ = function(width, height, data) {
const result = {width, height, ifd: this.ifd_, data};
this.sendResponse_(new LoadImageResponse(
LoadImageResponseStatus.SUCCESS, this.getClientTaskId(), result));
};
/**
* Handler, when contents are loaded into the image element. Performs image
* processing operations if needed, and finalizes the request process.
* @private
*/
ImageRequestTask.prototype.onImageLoad_ = function() {
const requestOrientation = this.request_.orientation;
// Override the request orientation before processing if needed.
if (this.renderOrientation_) {
this.request_.orientation = this.renderOrientation_;
}
// Perform processing if the url is not a data url, or if there are some
// operations requested.
let imageChanged = false;
if (!(this.request_.url.match(/^data/) ||
this.request_.url.match(/^drivefs:/)) ||
ImageLoaderUtil.shouldProcess(
this.image_.width, this.image_.height, this.request_)) {
ImageLoaderUtil.resizeAndCrop(this.image_, this.canvas_, this.request_);
imageChanged = true; // The image is now on the <canvas>.
}
// Restore the request orientation after processing.
if (this.renderOrientation_) {
this.request_.orientation = requestOrientation;
}
// Finalize the request.
this.sendImage_(imageChanged);
this.cleanup_();
this.downloadCallback_();
};
/**
* Handler, when loading of the image fails. Sends a failure response and
* finalizes the request process.
* @private
*/
ImageRequestTask.prototype.onImageError_ = function() {
this.sendResponse_(new LoadImageResponse(
LoadImageResponseStatus.ERROR, this.getClientTaskId()));
this.cleanup_();
this.downloadCallback_();
};
/**
* Cancels the request.
*/
ImageRequestTask.prototype.cancel = function() {
this.cleanup_();
// If downloading has started, then call the callback.
if (this.downloadCallback_) {
this.downloadCallback_();
}
};
/**
* Cleans up memory used by this request.
* @private
*/
ImageRequestTask.prototype.cleanup_ = function() {
this.image_.onerror = function() {};
this.image_.onload = function() {};
// Transparent 1x1 pixel gif, to force garbage collecting.
this.image_.src = '' +
'ABAAEAAAICTAEAOw==';
this.aborted_ = true;
if (this.xhr_) {
this.xhr_.abort();
}
// Dispose memory allocated by Canvas.
this.canvas_.width = 0;
this.canvas_.height = 0;
};