// ========================================================================== // Project: SproutCore
- JavaScript Application Framework // Copyright: ©2006-2011 Strobe Inc. and contributors. // Portions ©2008-2011 Apple Inc. All rights reserved. // Portions ©2010 Strobe Inc. // License: Licensed under MIT license (see license.js) // ==========================================================================
SC
.IMAGE_STATE_NONE = 'none'; SC
.IMAGE_STATE_LOADING = 'loading'; SC
.IMAGE_STATE_LOADED = 'loaded'; SC
.IMAGE_STATE_FAILED = 'failed';
SC
.IMAGE_TYPE_NONE = 'NONE'; SC
.IMAGE_TYPE_URL = 'URL'; SC
.IMAGE_TYPE_CSS_CLASS = 'CSS_CLASS';
/**
URL to a transparent GIF. Used for spriting.
*/ SC
.BLANK_IMAGE_DATAURL = “”;
SC
.BLANK_IMAGE_URL = SC.browser.isIE && SC.browser.compare(SC.browser.version, '8') <= 0 ? sc_static('blank.gif') : SC
.BLANK_IMAGE_DATAURL;
SC
.BLANK_IMAGE = new Image(); SC
.BLANK_IMAGE.src = SC
.BLANK_IMAGE_URL; SC
.BLANK_IMAGE.width = SC
.BLANK_IMAGE.height = 1;
/**
@class Displays an image in the browser. The ImageView can be used to efficiently display images in the browser. It includes a built in support for a number of features that can improve your page load time if you use a lot of images including a image loading cache and automatic support for CSS spriting. Note that there are actually many controls that will natively include images using an icon property name. @extends SC.View @extends SC.Control @extends SC.InnerFrame @since SproutCore 1.0
*/ SC
.ImageView = SC
.View.extend(SC
.Control, SC
.InnerFrame, /** @scope SC
.ImageView.prototype */ {
classNames: 'sc-image-view', // Don't apply this role until each image view can assign a non-empty string value for @aria-label <rdar://problem/9941887> // ariaRole: 'img', displayProperties: ['align', 'scale', 'value', 'displayToolTip'], /** @private */ renderDelegateName: function () { return (this.get('useCanvas') ? 'canvasImage' : 'image') + "RenderDelegate"; }.property('useCanvas').cacheable(), /** @private */ tagName: function () { return this.get('useCanvas') ? 'canvas' : 'div'; }.property('useCanvas').cacheable(), // .......................................................... // Properties // /** If YES, this image can load in the background. Otherwise, it is treated as a foreground image. If the image is not visible on screen, it will always be treated as a background image. */ canLoadInBackground: NO, /** @private @type Image @default SC.BLANK_IMAGE */ image: SC.BLANK_IMAGE, /** @private The frame for the inner img element or for the canvas to draw within, altered according to the scale and align properties provided by SC.InnerFrame. @type Object */ innerFrame: function () { var image = this.get('image'), imageWidth = image.width, imageHeight = image.height, frame = this.get('frame'); if (SC.none(frame)) return { x: 0, y: 0, width: 0, height: 0 }; // frame is 'null' until rendered when useStaticLayout === YES return this.innerFrameForSize(imageWidth, imageHeight, frame.width, frame.height); }.property('align', 'image', 'scale', 'frame').cacheable(), /** If YES, any specified toolTip will be localized before display. @type Boolean @default YES */ localize: YES, /** Current load status of the image. This status changes as an image is loaded from the server. If spriting is used, this will always be loaded. Must be one of the following constants: SC.IMAGE_STATE_NONE, SC.IMAGE_STATE_LOADING, SC.IMAGE_STATE_LOADED, SC.IMAGE_STATE_FAILED @type String */ status: SC.IMAGE_STATE_NONE, /** Will be one of the following constants: SC.IMAGE_TYPE_URL or SC.IMAGE_TYPE_CSS_CLASS @type String @observes value */ type: function () { var value = this.get('value'); if (SC.ImageView.valueIsUrl(value)) return SC.IMAGE_TYPE_URL; else if (!SC.none(value)) return SC.IMAGE_TYPE_CSS_CLASS; return SC.IMAGE_TYPE_NONE; }.property('value').cacheable(), /** The canvas element performs better than the img element since we can update the canvas image without causing browser reflow. As an additional benefit, canvas images are less easily copied, which is generally in line with acting as an 'application'. @type Boolean @default YES if supported @since SproutCore 1.5 */ useCanvas: function () { return SC.platform.supportsCanvas && !this.get('useStaticLayout'); }.property('useStaticLayout').cacheable(), /** If YES, image view will use the SC.imageQueue to control loading. This setting is generally preferred. @type Boolean @default YES */ useImageQueue: YES, /** A url or CSS class name. This is the image you want the view to display. It should be either a url or css class name. You can also set the content and contentValueKey properties to have this value extracted automatically. If you want to use CSS spriting, set this value to a CSS class name. If you need to use multiple class names to set your icon, separate them by spaces. Note that if you provide a URL, it must contain at least one '/' as this is how we autodetect URLs. @type String */ value: null, valueBindingDefault: SC.Binding.oneWay(), /** Recalculate our innerFrame if the outer frame has changed. @returns {void} */ viewDidResize: function () { sc_super(); // Note: SC.View's updateLayer() will call viewDidResize() if useStaticLayout is true. The result of this // is that since our display depends on the frame, when the view or parent view resizes, viewDidResize // notifies that the frame has changed, so we update our view, which calls viewDidResize, which notifies // that the frame has changed, so we update our view, etc. in an infinite loop. if (this.get('useStaticLayout')) { if (this._updatingOnce) { this._updatingOnce = false; } else { // Allow a single update when the view resizes to avoid an infinite loop. this._updatingOnce = true; this.updateLayerIfNeeded(); } } else { this.updateLayerIfNeeded(); } }, // .......................................................... // Methods // /** @private */ init: function () { sc_super(); // Start loading the image immediately on creation. this._valueDidChange(); if (this.get('useImageCache') !== undefined) { //@if(debug) SC.warn("Developer Warning: %@ has useImageCache set, please set useImageQueue instead".fmt(this)); //@endif this.set('useImageQueue', this.get('useImageCache')); } }, // .......................................................... // Rendering // /** Called when the element is attached to the document. If the image uses static layout (i.e. we don't know the frame beforehand), then this method will call updateLayerIfNeeded in order to adjust the inner frame of the image according to its rendered frame. */ didAppendToDocument: function () { // If using static layout, we can still support image scaling and aligning, // but we need to do it post-render. if (this.get('useStaticLayout')) { // Call updateLayer manually, because we can't have innerFrame be a // display property. It causes an infinite loop with static layout. this.updateLayerIfNeeded(); } }, /** Called when the element is created. If the view is using a canvas element, then we can not draw to the canvas until it exists. This method will call updateLayerIfNeeded in order to draw to the canvas. */ didCreateLayer: function () { if (this.get('useCanvas')) { this.updateLayerIfNeeded(); } }, // .......................................................... // Value handling // /** @private Whenever the value changes, update the image state and possibly schedule an image to load. */ _valueDidChange: function () { var value = this.get('value'), type = this.get('type'); // Reset the backing image object every time. this.set('image', SC.BLANK_IMAGE); if (type == SC.IMAGE_TYPE_URL) { // Load the image. this.set('status', SC.IMAGE_STATE_LOADING); // order: image cache, normal load if (!this._loadImageUsingCache()) { this._loadImage(); } } }.observes('value'), /** @private Tries to load the image value using the SC.imageQueue object. If the value is not a URL, it won't attempt to load it using this method. @returns YES if loading using SC.imageQueue, NO otherwise */ _loadImageUsingCache: function () { var value = this.get('value'), type = this.get('type'); // now update local state as needed.... if (type === SC.IMAGE_TYPE_URL && this.get('useImageQueue')) { var isBackground = this.get('isVisibleInWindow') || this.get('canLoadInBackground'); SC.imageQueue.loadImage(value, this, this._loadImageUsingCacheDidComplete, isBackground); return YES; } return NO; }, /** @private */ _loadImageUsingCacheDidComplete: function (url, image) { var value = this.get('value'); if (value === url) { if (SC.ok(image)) { this.didLoad(image); } else { // if loading it using the cache didn't work, it's useless to try loading the image normally this.didError(image); } } }, /** @private Loads an image using a normal Image object, without using the SC.imageQueue. @returns YES if it will load, NO otherwise */ _loadImage: function () { var value = this.get('value'), type = this.get('type'), that = this, image, jqImage; if (type === SC.IMAGE_TYPE_URL) { image = new Image(); var errorFunc = function () { SC.run(function () { that._loadImageDidComplete(value, SC.$error("SC.Image.FailedError", "Image", -101)); }); }; var loadFunc = function () { SC.run(function () { that._loadImageDidComplete(value, image); }); }; // Don't grab the jQuery object repeatedly jqImage = $(image); // Using bind here instead of setting onabort/onerror/onload directly // fixes an issue with images having 0 width and height jqImage.bind('error', errorFunc); jqImage.bind('abort', errorFunc); jqImage.bind('load', loadFunc); image.src = value; return YES; } return NO; }, /** @private */ _loadImageDidComplete: function (url, image) { var value = this.get('value'); if (value === url) { if (SC.ok(image)) { this.didLoad(image); } else { this.didError(image); } } }, didLoad: function (image) { this.set('status', SC.IMAGE_STATE_LOADED); if (!image) image = SC.BLANK_IMAGE; this.set('image', image); this.displayDidChange(); }, didError: function (error) { this.set('status', SC.IMAGE_STATE_FAILED); this.set('image', SC.BLANK_IMAGE); }
});
/**
Returns YES if the passed value looks like an URL and not a CSS class name.
*/ SC
.ImageView.valueIsUrl = function (value) {
return value && SC.typeOf(value) === SC.T_STRING ? value.indexOf('/') >= 0 : NO;
};