/* global Ext */
/*******************************************************************************
* This file is distributed on an AS IS BASIS WITHOUT ANY WARRANTY; without even
* the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
* ***********************************************************************************
*
* License: ux.ManagedIFrame and ux.ManagedIFramePanel 1.2 are licensed under
* the terms of the Open Source LGPL 3.0 license:
* http://www.gnu.org/licenses/lgpl.html
*
* Commercial use is prohibited without a Commercial License. See
* http://licensing.theactivegroup.com.
*
* Donations are welcomed: http://donate.theactivegroup.com
*
*
* An Ext.Element harness for iframe elements.
*
* Adds Ext.UpdateManager(Updater) support and a compatible 'update' method for
* writing content directly into an iFrames' document structure.
*
* Signals various DOM/document states as the frames content changes with
* 'domready', 'documentloaded', and 'exception' events. The domready event is
* only raised when a proper security context exists for the frame's DOM to
* permit modification. (ie, Updates via Updater or documents retrieved from
* same-domain servers).
*
* Frame sand-box permits eval/script-tag writes of javascript source. (See
* execScript, writeScript, and loadFunction methods for more info.)
*
* Usage:
*
*
support (unsupportedText)
* Mod: Improved domready detection for Opera, Webkit, and IE. jsDOC updates.
* Mod: MIFP frameConfig now honors either frameConfig:{id:'someId': name:'someName',....}
* or frameConfig: {autoCreate:{ id:'someId': name:'someName',....}}
* Add: resetUrl class property to override the reset URL used by default with the reset method.
* Add: (MIF/MIFP):setLocation method. Identical to setSrc, but uses location.replace
* on the frame instead, preventing a History update.
* Add: scope parameter to
* all methods supporting callbacks
* Mod: callbacks are invoke as soon as domready status is detected.
* Mod: MIFPanel now honors the contentEl cfg option, meaning IFRAMEs authored in page markup
* may be converted to MIFrames when the Panel is rendered.
*
* 1.2 (8/22/2008) FF3 Compatibility Fixes, loadMask tweaks
*
* 1.1 (4/13/2008) Adds Ext.Element, CSS Selectors (query,select) fly, and CSS
* interface support (same-domain only) Adds blur,focus,unload events
* (same-domain only)
*
*/
(function() {
var addListener = function () {
if (window.addEventListener) {
return function(el, eventName, fn, capture) {
el.addEventListener(eventName, fn, !!capture);
};
} else if (window.attachEvent) {
return function(el, eventName, fn, capture) {
el.attachEvent("on" + eventName, fn);
};
} else {
return function() {
};
}
}(),
removeListener = function() {
if (window.removeEventListener) {
return function (el, eventName, fn, capture) {
el.removeEventListener(eventName, fn, (capture));
};
} else if (window.detachEvent) {
return function (el, eventName, fn) {
el.detachEvent("on" + eventName, fn);
};
} else {
return function() {
};
}
}();
var EV = Ext.lib.Event;
var MIM;
var MASK_TARGET = 'x-frame-mask-target';
/**
* @class Ext.ux.ManagedIFrame
* @extends Ext.Element
* @extends Ext.util.Observable
* @version: 1.2.7 (10/02/2009)
* @license LGPL 3.0
* @author: Doug Hendricks. Forum ID: hendricd
* @donate
*
* @cfg {Boolean/Object} autoCreate True to auto generate the IFRAME element, or a {@link Ext.DomHelper} config of the IFRAME to create
* @cfg {String} html Any markup to be applied to the IFRAME's document content when rendered.
* @cfg {Object} loadMask An {@link Ext.LoadMask} config or true to mask the iframe while using the update or setSrc methods (defaults to false).
* @cfg {Object} src The src attribute to be assigned to the Iframe after initialization (overrides the autoCreate config src attribute)
* @constructor
* @param {Mixed} el, Config object The iframe element or it's id to harness or a valid config object.
*/
Ext.ux.ManagedIFrame = function() {
var args = Array.prototype.slice.call(arguments, 0),
el = Ext.get(args[0]),
config = args[0];
if (el && el.dom && el.dom.tagName == 'IFRAME') {
config = args[1] || {};
} else {
config = args[0] || args[1] || {};
el = config.autoCreate ? Ext.get(Ext.DomHelper.append(
config.autoCreate.parent || Ext.getBody(), Ext.apply({
tag : 'iframe',
frameborder : 0,
src : (Ext.isIE && Ext.isSecure)? Ext.SSL_SECURE_URL: 'about:blank'
}, config.autoCreate)))
: null;
if(el && this.unsupportedText){
Ext.DomHelper.append(el.dom.parentNode, {tag:'noframes',html:this.unsupportedText } );
}
}
if (!el || el.dom.tagName != 'IFRAME') { return el; }
el.dom.name || (el.dom.name = el.dom.id); // make sure there is a valid frame name
el.dom.ownerEl = el;
this.addEvents({
/**
* Fires when the frame gets focus. Note: This event is only
* available when overwriting the iframe document using the
* {@link #Ext.ux.ManagedIFrame-update} method and to pages
* retrieved from a "same-origin" domain. Returning false from the
* eventHandler [MAY] NOT cancel the event, as this event is NOT
* ALWAYS cancellable in all browsers.
*
* @event focus
* @param {Ext.ux.ManagedIFrame} this
* @param {Ext.Event} event
*
*/
"focus" : true,
/**
* Fires when the frame is blurred (loses focus). Note: This event
* is only available when overwriting the iframe document using the
* update method and to pages retrieved from a "same-origin" domain.
* Returning false from the eventHandler [MAY] NOT cancel the event,
* as this event is NOT ALWAYS cancellable in all browsers.
*
* @event blur
* @param {Ext.ux.ManagedIFrame} this
* @param {Ext.Event} event
*/
"blur" : true,
/**
* Note: This event is only available when overwriting the iframe
* document using the {@link #Ext.ux.ManagedIFrame-update} method
* and to documents retrieved from a "same-origin" domains. Fires
* when(if) the frames window object raises the unload event
* Note: Opera does not raise this event.
*
* @event unload
* @param {Ext.ux.ManagedIFrame} this
* @param {Ext.Event} event
*/
"unload" : true,
/**
* Note: This event is only available when overwriting the iframe
* document using the {@link #Ext.ux.ManagedIFrame-update} method
* and to documents retrieved from a "same-origin" domains. Fires
* ONLY when an iFrame's Document(DOM) has reach a state where the
* DOM may be manipulated (ie same domain policy) Returning false
* from the eventHandler stops further event (documentloaded)
* processing.
*
* @event domready
* @param {Ext.ux.ManagedIFrame} this
*/
"domready" : true,
/**
* Fires when the iFrame has reached a loaded/complete state.
*
* @event documentloaded
* @param {Ext.ux.ManagedIFrame} this
*/
"documentloaded" : true,
/**
* Fires when the frame actions raise an error
*
* @event exception
* @param {Ext.ux.ManagedIFrame} this
* @param {Error|string} exception
*/
"exception" : true,
/**
* Fires upon receipt of a message generated by window.sendMessage
* method of the embedded Iframe.window object
*
* @event message
* @param {Ext.ux.ManagedIFrame} this
* @param {Object} message members:
*
* - type {string}
*
* literal "message"
*
*
* - data {Mixed}
*
* the message payload]
*
*
* - domain {String}
*
* the document domain from which the message originated
*
*
* - uri {string}
*
* the document URI of the message sender
*
*
* - source (Object)
*
* the window context of the message sender
*
*
* - tag {string}
*
* optional reference tag sent by the message sender }
*
*
*
*/
"message" : true
// "message:tagName" is supported for X-frame messaging
/**
* Alternate event handler syntax for message:tag filtering
* Fires upon receipt of a message generated by
* window.sendMessage method which includes a specific tag value
* of the embedded Iframe.window object
*
* @event message:tag
* @param {Ext.ux.ManagedIFrame} this
* @param {Object} message members:
*
* - type {string}
*
* literal "message"
*
*
* - data {Mixed}
*
* the message payload]
*
*
* - domain {String}
*
* the document domain from which the message
* originated
*
*
* - uri {string}
*
* the document URI of the message sender
*
*
* - source (Object)
*
* the window context of the message sender
*
*
* - tag {string}
*
* optional reference tag sent by the message sender }
*
*
*
*/
});
if (config.listeners) {
this.listeners = config.listeners;
Ext.ux.ManagedIFrame.superclass.constructor.call(this);
}
Ext.apply(el, this); // apply this class interface ( pseudo Decorator
// )
el.addClass('x-managed-iframe');
if (config.style) {
el.applyStyles(config.style);
}
Ext.apply(el, {
disableMessaging : config.disableMessaging === true,
loadMask : !!config.loadMask ? Ext.apply({
msg : 'Loading..',
maskEl : null,
hideOnReady : false,
disabled : false
}, config.loadMask) : false
,
_windowContext : null,
eventsFollowFrameLinks : typeof config.eventsFollowFrameLinks == 'undefined'
? true : config.eventsFollowFrameLinks
});
if(el.loadMask ){
el.loadMask.maskEl || (el.loadMask.maskEl = el.parent('.'+MASK_TARGET) || el.parent());
el.loadMask.maskEl.addClass(MASK_TARGET);
}
var um = el.updateManager = new Ext.UpdateManager(el, true);
um.showLoadIndicator = config.showLoadIndicator || false;
Ext.ux.ManagedIFrame.Manager.register(el);
if (config.src) {
el.setSrc(config.src);
} else {
var content = config.html || config.content || false;
if (content) {
el.reset(null,function(frame){
// permit the syntax for supported arguments: content or [content, loadScripts, callback, scope]
frame.update.apply(el, [].concat(content)); });
}
}
return el;
};
Ext.extend(Ext.ux.ManagedIFrame, Ext.util.Observable, {
/** (read-only) The last known URI set programmatically by the Component
* @cfg {String|Function} src The target URI of the Component to set after render.
* @property
* @type String
*/
src : null,
/** (read-only) For "same-origin" frames only. Provides a reference to
* the Ext.util.CSS singleton to manipulate the style sheets of the frame's
* embedded document.
*
* @property
* @type Ext.util.CSS
*/
CSS : null,
/** Provides a reference to the managing Ext.ux.ManagedIFrame.Manager instance.
*
* @property
* @type Ext.ux.ManagedIFrame.Manager
*/
manager : null,
/**
* Enables/disables internal cross-frame messaging interface
* @cfg {Boolean} disableMessaging False to enable cross-frame messaging API
* Default = true
*
*/
disableMessaging : true,
/**
* Maximum number of domready event detection retries for IE. IE does not provide
* a native DOM event to signal when the frames DOM may be manipulated, so a polling process
* is used to determine when the documents BODY is available. Certain documents may not contain
* a BODY tag: eg. MHT(rfc/822), XML, or other non-HTML content. Detection polling will stop after this number of 2ms retries
* or when the documentloaded event is raised.
* @cfg {Integer} domReadyRetries
* @default 7500 (* 2ms = 15 seconds)
*/
domReadyRetries : 7500,
/**
* @cfg focusOnLoad True to set focus on the frame Window as soon as its document
* reports loaded. (Many external sites use IE's document.createRange to create
* DOM elements, but to be successfull IE requires that the FRAME have focus before
* the method is called)
* @default false
*/
focusOnLoad : false,
/**
* @cfg {String} resetUrl Frame document reset string for use with the {@link #Ext.ux.ManagedIFrame-reset} method.
* Defaults: For IE on SSL domains - the current value of Ext.SSL_SECURE_URL
"about:blank" for all others.
*/
resetUrl : (function(){
if(Ext.isIE && Ext.isSecure){
return Ext.SSL_SECURE_URL;
} else {
return 'about:blank';
}
})(),
/**
* @cfg {String} unsupportedText Text to display when the IFRAMES.FRAMESETS are disabled by the browser.
*
*/
unsupportedText : 'Inline frames are NOT enabled\/supported by your browser.',
/**
* Sets the embedded Iframe src property. Note: invoke the function with
* no arguments to refresh the iframe based on the current src value.
*
* @param {String/Function} url (Optional) A string or reference to a Function that
* returns a URI string when called
* @param {Boolean} discardUrl (Optional) If not passed as false
* the URL of this action becomes the default SRC attribute
* for this iframe, and will be subsequently used in future
* setSrc calls (emulates autoRefresh by calling setSrc
* without params).
* @param {Function} callback (Optional) A callback function invoked when the
* frame document has been fully loaded.
* @param {Object} scope (Optional) scope by which the callback function is
* invoked.
*/
setSrc : function(url, discardUrl, callback, scope) {
if (url && typeof url == 'object') {
callback = url.callback || false;
discardUrl = url.discardUrl || false;
url = url.url || false;
scope = url.scope || null;
}
var src = url || this.src || this.resetUrl;
this._windowContext = null;
this._unHook();
this._frameAction = this.frameInit = this._domReady = false;
this.showMask();
var s = this._targetURI = typeof src == 'function' ? src() || '' : src;
try {
this._frameAction = true; // signal listening now
this._callBack = typeof callback == 'function' ? callback.createDelegate(scope) : null;
this.dom.src = s;
this.frameInit = true; // control initial event chatter
this.checkDOM();
} catch (ex) {
this.fireEvent('exception', this, ex);
}
if (discardUrl !== true) {
this.src = src;
}
return this;
},
/**
* Sets the embedded Iframe location using its replace method. Note: invoke the function with
* no arguments to refresh the iframe based on the current src value.
*
* @param {String/Function} url (Optional) A string or reference to a Function that
* returns a URI string when called
* @param {Boolean} discardUrl (Optional) If not passed as false
* the URL of this action becomes the default SRC attribute
* for this iframe, and will be subsequently used in future
* setSrc calls (emulates autoRefresh by calling setSrc
* without params).
* @param {Function} callback (Optional) A callback function invoked when the
* frame document has been fully loaded.
* @param {Object} scope (Optional) scope by which the callback function is
* invoked.
*
*/
setLocation : function(url, discardUrl, callback, scope) {
if (url && typeof url == 'object') {
callback = url.callback || false;
discardUrl = url.discardUrl || false;
url = url.url || false;
scope = url.scope || null;
}
var src = url || this.src || this.resetUrl;
this._windowContext = null;
this._unHook();
this._frameAction = this.frameInit = this._domReady = false;
this.showMask();
var s = this._targetURI = typeof src == 'function' ? src() || '' : src;
try {
this._frameAction = true; // signal listening now
this._callBack = typeof callback == 'function' ? callback.createDelegate(scope) : null;
this.getWindow().location.replace(s);
this.frameInit = true; // control initial event chatter
this.checkDOM();
} catch (ex) {
this.fireEvent('exception', this, ex);
}
if (discardUrl !== true) {
this.src = src;
}
return this;
},
/**
* Resets the frame to a neutral (blank document) state without
* loadMasking.
*
* @param {String}
* src (Optional) A specific reset string (eg. 'about:blank')
* to use for resetting the frame.
* @param {Function}
* callback (Optional) A callback function invoked when the
* frame reset is complete.
* @param {Object}
* scope (Optional) scope by which the callback function is
* invoked.
*/
reset : function(src, callback, scope) {
this._unHook();
var loadMaskOff = false;
if(this.loadMask){
loadMaskOff = this.loadMask.disabled;
this.loadMask.disabled = false;
}
this._callBack = function(frame) {
if(frame.loadMask){
frame.loadMask.disabled = loadMaskOff;
};
frame._frameAction = false;
frame.frameInit = true;
this._isReset= false;
if (callback) {
callback.call(scope || window, frame);
}
};
this.hideMask(true);
this._frameAction = false; // no chatter on reset
this.frameInit = true
this._isReset= true;
var s = src;
if (typeof src == 'function') { s = src();}
s = this._targetURI = Ext.isEmpty(s, true)? this.resetUrl: s;
this.getWindow().location.href = s;
return this;
},
/**
* Regular Expression filter pattern for script tag removal.
* @cfg {regexp} scriptRE script removal RegeXp
* Default: "/(?:)((\n|\r|.)*?)(?:<\/script>)/gi"
*/
scriptRE : /(?:)((\n|\r|.)*?)(?:<\/script>)/gi,
/**
* Write(replacing) string content into the IFrames document structure
* @param {String} content The new content
* @param {Boolean} loadScripts
* (optional) true to also render and process embedded scripts
* @param {Function} callback (Optional) A callback function invoked when the
* frame document has been written and fully loaded. @param {Object}
* scope (Optional) scope by which the callback function is invoked.
*/
update : function(content, loadScripts, callback, scope) {
loadScripts = loadScripts || this.getUpdateManager().loadScripts || false;
content = Ext.DomHelper.markup(content || '');
content = loadScripts === true ? content : content.replace(this.scriptRE, "");
var doc;
if ((doc = this.getDocument()) && !!content.length) {
this._unHook();
this._windowContext = this.src = null;
this._targetURI = location.href;
this.src = null;
this.frameInit = true; // control initial event chatter
this.showMask();
this._callBack = typeof callback == 'function' ? callback.createDelegate(scope) : null;
doc.open();
this._frameAction = true;
doc.write(content);
doc.close();
this.checkDOM();
} else {
this.hideMask(true);
if (callback) {
callback.call(scope, this);
}
}
return this;
},
/**
* Enables/disables internal cross-frame messaging interface
*
* @cfg {Boolean} disableMessaging False to enable cross-frame messaging
* API (default is True)
*/
disableMessaging : true,
/**
* @private, frame messaging interface (for same-domain-policy frames
* only)
*/
_XFrameMessaging : function() {
// each tag gets a hash queue ($ = no tag ).
var tagStack = { '$' : [] };
var isEmpty = function(v, allowBlank) {
return v === null || v === undefined
|| (!allowBlank ? v === '' : false);
};
var apply = function(o, c, defaults) {
if (defaults) { apply(o, defaults); }
if (o && c && typeof c == 'object') {
for (var p in c) {
o[p] = c[p];
}
}
return o;
};
window.sendMessage = function(message, tag, origin) {
var MIF;
if (MIF = arguments.callee.manager) {
if (message._fromHost) {
var fn, result;
// only raise matching-tag handlers
var compTag = message.tag || tag || null;
var mstack = !isEmpty(compTag) ? tagStack[compTag.toLowerCase()] || [] : tagStack["$"];
for (var i = 0, l = mstack.length; i < l; i++) {
if (fn = mstack[i]) {
result = fn.apply(fn.__scope, arguments) === false? false: result;
if (fn.__single) {
mstack[i] = null;
}
if (result === false) {
break;
}
}
}
return result;
} else {
message = {
type : isEmpty(tag) ? 'message' : 'message:'
+ tag.toLowerCase().replace(/^\s+|\s+$/g,''),
data : message,
domain : origin || document.domain,
uri : document.documentURI,
source : window,
tag : isEmpty(tag) ? null : tag.toLowerCase()
};
try {
return MIF.disableMessaging !== true
? MIF.fireEvent.call(MIF, message.type,MIF, message)
: null;
} catch (ex) {
} // trap for message:tag handlers not yet defined
return null;
}
}
};
window.onhostmessage = function(fn, scope, single, tag) {
if (typeof fn == 'function') {
if (!isEmpty(fn.__index)) {
throw "onhostmessage: duplicate handler definition"
+ (tag ? " for tag:" + tag : '');
}
var k = isEmpty(tag) ? "$" : tag.toLowerCase();
tagStack[k] || (tagStack[k] = []);
apply(fn, {
__tag : k,
__single : single || false,
__scope : scope || window,
__index : tagStack[k].length
});
tagStack[k].push(fn);
} else {
throw "onhostmessage: function required";
}
};
window.unhostmessage = function(fn) {
if (typeof fn == 'function' && typeof fn.__index != 'undefined') {
var k = fn.__tag || "$";
tagStack[k][fn.__index] = null;
}
};
},
/**
* Method to retrieve frame's history object.
* @return {object} or null if permission was denied
*/
getHistory : function(){
var h=null;
try{ h=this.getWindow().history; }catch(eh){}
return h;
},
/**
* Method to retrieve embedded frame Element objects. Uses simple
* caching (per frame) to consistently return the same object.
* Automatically fixes if an object was recreated with the same id via
* AJAX or DOM.
*
* @param {Mixed}
* el The id of the node, a DOM Node or an existing Element.
* @return {Element} The Element object (or null if no matching element
* was found)
*/
get : function(el) {
return MIM.El.get(this, el);
},
/**
* Gets the globally shared flyweight Element for the frame, with the
* passed node as the active element. Do not store a reference to this
* element - the dom node can be overwritten by other code.
*
* @param {String/HTMLElement}
* el The dom node or id
* @param {String}
* named (optional) Allows for creation of named reusable
* flyweights to prevent conflicts (e.g. internally Ext uses
* "_internal")
* @return {Element} The shared Element object (or null if no matching
* element was found)
*/
fly : function(el, named) {
named = named || '_global';
el = this.getDom(el);
if (!el) {
return null;
}
if (!MIM._flyweights[named]) {
MIM._flyweights[named] = new Ext.Element.Flyweight();
}
MIM._flyweights[named].dom = el;
return MIM._flyweights[named];
},
/**
* Return the dom node for the passed string (id), dom node, or
* Ext.Element relative to the embedded frame document context.
*
* @param {Mixed} el
* @return HTMLElement
*/
getDom : function(el) {
var d;
if (!el || !(d = this.getDocument())) {
return null;
}
return el.dom ? el.dom : (typeof el == 'string' ? d
.getElementById(el) : el);
},
/**
* Creates a {@link Ext.CompositeElement} for child nodes based on the
* passed CSS selector (the selector should not contain an id).
*
* @param {String} selector The CSS selector
* @param {Boolean} unique (optional) True to create a unique Ext.Element for
* each child (defaults to false, which creates a single
* shared flyweight object)
* @return {Ext.CompositeElement/Ext.CompositeElementLite} The composite element
*/
select : function(selector, unique) {
var d;
return (d = this.getDocument()) ? Ext.Element.select(selector,unique, d) : null;
},
/**
* Selects frame document child nodes based on the passed CSS selector
* (the selector should not contain an id).
*
* @param {String} selector The CSS selector
* @return {Array} An array of the matched nodes
*/
query : function(selector) {
var d;
return (d = this.getDocument()) ? Ext.DomQuery.select(selector, d): null;
},
/**
* Returns the frame's current HTML document object as an
* {@link Ext.Element}.
*
* @return {Ext.Element} The document
*/
getDoc : function() {
return this.get(this.getDocument());
},
/**
* Removes a DOM Element from the embedded documents
*
* @param {Element/String} node The node id or node Element to remove
*
*/
removeNode : function(node) {
MIM.removeNode(this, this.getDom(node));
},
/** @private : clear all event listeners and Element cache */
_unHook : function() {
var elcache, h = MIM.getFrameHash(this) || {};
if (this._hooked){
if( h && (elcache = h.elCache)) {
for (var id in elcache) {
var el = elcache[id];
if (el.removeAllListeners) {
el.removeAllListeners();
}
delete elcache[id];
}
if (h.docEl) {
h.docEl.removeAllListeners();
h.docEl = null;
delete h.docEl;
}
}
var w;
if(this._frameProxy && (w = this.getWindow())){
removeListener(w, 'focus', this._frameProxy);
removeListener(w, 'blur', this._frameProxy);
removeListener(w, 'resize', this._frameProxy);
removeListener(w, 'unload', this._frameProxy);
}
}
this._hooked = this._domReady = this._domFired = this._frameAction = false;
MIM._flyweights = {};
this.CSS = this.CSS ? this.CSS.destroy() : null;
},
// Private execScript sandbox and messaging interface
_renderHook : function() {
this._windowContext = null;
this.CSS = this.CSS ? this.CSS.destroy() : null;
this._hooked = false;
try {
if (this.writeScript('(function(){(window.hostMIF = parent.Ext.get("'
+ this.dom.id
+ '"))._windowContext='
+ (Ext.isIE
? 'window'
: '{eval:function(s){return eval(s);}}')
+ ';})();')) {
this._frameProxy || (this._frameProxy = MIM.eventProxy.createDelegate(this));
var w;
if(w = this.getWindow()){
addListener(w, 'focus', this._frameProxy);
addListener(w, 'blur', this._frameProxy);
addListener(w, 'resize', this._frameProxy);
addListener(w, 'unload', this._frameProxy);
}
if (this.disableMessaging !== true) {
this.loadFunction({
name : 'XMessage',
fn : this._XFrameMessaging
}, false, true);
var sm;
if (sm = w.sendMessage) {
sm.manager = this;
}
}
this.CSS = new CSSInterface(this.getDocument());
}
} catch (ex) {}
return (this._hooked = this.domWritable());
},
/**
* dispatch a message to the embedded frame-window context
* @name sendMessage
* @methodOf Ext.ux.ManagedIFrame
* @param {Mixed} message The message payload
* @param {String} tag Optional reference tag
* @param {String} origin Optional domain designation of the sender (defaults
* to document.domain).
*/
sendMessage : function(message, tag, origin) {
var win;
if (this.disableMessaging !== true && (win = this.getWindow())) {
// support frame-to-frame messaging relay
tag || (tag = message.tag || '');
tag = tag.toLowerCase();
message = Ext.applyIf(message.data ? message : {
data : message
}, {
type : Ext.isEmpty(tag) ? 'message' : 'message:'
+ tag,
domain : origin || document.domain,
uri : document.documentURI,
source : window,
tag : tag || null,
_fromHost : this
});
return win.sendMessage ? win.sendMessage.call(null, message,
tag, origin) : null;
}
return null;
},
/** @private */
_windowContext : null,
/**
* If sufficient privilege exists, returns the frame's current document
* as an HTMLElement.
*
* @return {HTMLElement} The frame document or false if access to
* document object was denied.
*/
getDocument : function() {
var win = this.getWindow(), doc = null;
try {
doc = (Ext.isIE && win ? win.document : null)
|| this.dom.contentDocument
|| window.frames[this.id].document || null;
} catch (gdEx) {
return false; // signifies probable access restriction
}
return doc;
},
/**
* If sufficient privilege exists, returns the frame's current document
* body as an HTMLElement.
*
* @return {HTMLElement} The frame document body or Null if access to
* document object was denied.
*/
getBody : function() {
var d;
return (d = this.getDocument()) ? d.body : null;
},
/**
* Attempt to retrieve the frames current URI via frame's document object
* @return {string} The frame document's current URI or the last know URI if permission was denied.
*/
getDocumentURI : function() {
var URI, d;
try {
URI = this.src && (d = this.getDocument()) ? d.location.href: null;
} catch (ex) { // will fail on NON-same-origin domains
}
return URI || (typeof this.src == 'function' ? this.src() : this.src);
// fallback to last known
},
/**
* Attempt to retrieve the frames current URI via frame's Window object
* @return {string} The frame document's current URI or the last know URI if permission was denied.
*/
getWindowURI : function() {
var URI, w;
try {
URI = (w = this.getWindow()) ? w.location.href : null;
} catch (ex) {
} // will fail on NON-same-origin domains
return URI || (typeof this.src == 'function' ? this.src() : this.src);
// fallback to last known
},
/**
* Returns the frame's current window object.
*
* @return {Window} The frame Window object.
*/
getWindow : function() {
var dom = this.dom, win = null;
try {
win = dom.contentWindow || window.frames[dom.name] || null;
} catch (gwEx) {
}
return win;
},
/**
* Print the contents of the Iframes (if we own the document)
*/
print : function() {
var win;
try {
if( win = this.getWindow()){
if (Ext.isIE) {
win.focus();
}
win.print();
}
} catch (ex) {
throw 'print exception: ' + (ex.description || ex.message || ex);
}
},
/** @private */
destroy : function() {
this.removeAllListeners();
if (this.loadMask) {
this.hideMask(true);
Ext.apply(this.loadMask, {
masker : null,
maskEl : null
});
}
if (this.dom) {
Ext.ux.ManagedIFrame.Manager.deRegister(this);
this.dom.ownerEl = this._windowContext = null;
// IE Iframe cleanup
if (Ext.isIE && this.dom.src) {
this.dom.src = 'javascript:false';
}
this._maskEl = null;
this.remove();
}
},
/**
* Returns the general DOM modification capability of the frame. @return
* {Boolean} If True, the frame's DOM can be manipulated, queried, and
* Event Listeners set.
*/
domWritable : function() {
return !!this._windowContext;
},
/**
* eval a javascript code block(string) within the context of the
* Iframes window object.
* @param {String} block A valid ('eval'able) script source block.
* @param {Boolean} useDOM if true, inserts the function
* into a dynamic script tag, false does a simple eval on the function
* definition. (useful for debugging) Note: will only work after a
* successful iframe.(Updater) update or after same-domain document has
* been hooked, otherwise an exception is raised.
*/
execScript : function(block, useDOM) {
try {
if (this.domWritable()) {
if (useDOM) {
this.writeScript(block);
} else {
return this._windowContext.eval(block);
}
} else {
throw 'execScript:non-secure context'
}
} catch (ex) {
this.fireEvent('exception', this, ex);
return false;
}
return true;
},
/**
* write a