2308 lines
80 KiB
Plaintext
2308 lines
80 KiB
Plaintext
|
/*
|
||
|
* Copyright 2005 Joe Walker
|
||
|
*
|
||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||
|
* you may not use this file except in compliance with the License.
|
||
|
* You may obtain a copy of the License at
|
||
|
*
|
||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||
|
*
|
||
|
* Unless required by applicable law or agreed to in writing, software
|
||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||
|
* See the License for the specific language governing permissions and
|
||
|
* limitations under the License.
|
||
|
*/
|
||
|
|
||
|
/**
|
||
|
* The DWR object is also defined by dwr.util etc.
|
||
|
*/
|
||
|
if (typeof dojo != 'undefined') dojo.provide('dwr.engine');
|
||
|
if (typeof dwr == 'undefined') dwr = {};
|
||
|
|
||
|
(function() {
|
||
|
dwr.engine = {};
|
||
|
|
||
|
/**
|
||
|
* Set an alternative error handler from the default alert box.
|
||
|
* @param {Function} handler The function to call when an error happens
|
||
|
* @see http://getahead.org/dwr/browser/engine/errors
|
||
|
*/
|
||
|
dwr.engine.setErrorHandler = function(handler) {
|
||
|
dwr.engine._errorHandler = handler;
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* Set an alternative warning handler from the default alert box.
|
||
|
* @param {Function} handler The function to call when a warning happens
|
||
|
* @see http://getahead.org/dwr/browser/engine/errors
|
||
|
*/
|
||
|
dwr.engine.setWarningHandler = function(handler) {
|
||
|
dwr.engine._warningHandler = handler;
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* Setter for the text/html handler - what happens if a DWR request gets an HTML
|
||
|
* reply rather than the expected Javascript. Often due to login timeout
|
||
|
* @param {Function} handler The function to call on an unexpected text/html content type
|
||
|
*/
|
||
|
dwr.engine.setTextHtmlHandler = function(handler) {
|
||
|
dwr.engine._textHtmlHandler = handler;
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* Set a default timeout value for all calls. 0 (the default) turns timeouts off.
|
||
|
* @param {Function} handler The function to call when we get bored of waiting for a call
|
||
|
* @see getahead.org/dwr/browser/engine/errors
|
||
|
*/
|
||
|
dwr.engine.setTimeout = function(timeout) {
|
||
|
dwr.engine._timeout = timeout;
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* The Pre-Hook is called before any DWR remoting is done.
|
||
|
* @param {Function} handler The function to call before any remote calls
|
||
|
* @see getahead.org/dwr/browser/engine/hooks
|
||
|
*/
|
||
|
dwr.engine.setPreHook = function(handler) {
|
||
|
dwr.engine._preHook = handler;
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* The Post-Hook is called after any DWR remoting is done.
|
||
|
* @param {Function} handler The function to call after any remote calls
|
||
|
* @see getahead.org/dwr/browser/engine/hooks
|
||
|
*/
|
||
|
dwr.engine.setPostHook = function(handler) {
|
||
|
dwr.engine._postHook = handler;
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* Custom headers for all DWR calls
|
||
|
* @param {Object} headers Object containing name/value pairs for extra headers
|
||
|
* @see getahead.org/dwr/????
|
||
|
*/
|
||
|
dwr.engine.setHeaders = function(headers) {
|
||
|
dwr.engine._headers = headers;
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* @Deprecated - use setRequestAttributes
|
||
|
*/
|
||
|
dwr.engine.setParameters = function(parameters) {
|
||
|
dwr.engine.setRequestAttributes(parameters);
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* Custom request attributes for all DWR calls
|
||
|
* @param {Object} attributes Object containing name/value pairs for request attributes
|
||
|
* @see getahead.org/dwr/????
|
||
|
*/
|
||
|
dwr.engine.setRequestAttributes = function(attributes) {
|
||
|
dwr.engine._requestAttributes = attributes;
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* Ensure that remote calls happen in the order in which they were sent? (Default: false)
|
||
|
* @param {boolean} ordered true to enable ordered processing
|
||
|
* @see getahead.org/dwr/browser/engine/ordering
|
||
|
*/
|
||
|
dwr.engine.setOrdered = function(ordered) {
|
||
|
dwr.engine._ordered = ordered;
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* Do we ask the XHR object to be asynchronous? (Default: true)
|
||
|
* Warning: it is <strong>highly</strong> advised to use the default ofasync
|
||
|
* processing, especially when dealing with Internet based requests.
|
||
|
* @param {boolean} async false to enable sync processing for XHR queries
|
||
|
* @see getahead.org/dwr/browser/engine/options
|
||
|
*/
|
||
|
dwr.engine.setAsync = function(async) {
|
||
|
dwr.engine._async = async;
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* Does the client actively check the server for updates? (Default: false)
|
||
|
* @param {boolean} async true to enable low latency reverse ajax
|
||
|
* @see getahead.org/dwr/browser/engine/options
|
||
|
*/
|
||
|
dwr.engine.setActiveReverseAjax = function(activeReverseAjax) {
|
||
|
if (activeReverseAjax) {
|
||
|
// Bail if we are already started
|
||
|
if (dwr.engine._activeReverseAjax) return;
|
||
|
// We always want a retry policy when reverse AJAX is enabled.
|
||
|
dwr.engine._retryIntervals = dwr.engine._defaultRetryIntervals;
|
||
|
dwr.engine._activeReverseAjax = true;
|
||
|
dwr.engine._poll();
|
||
|
}
|
||
|
else {
|
||
|
// Can we cancel an existing request?
|
||
|
if (dwr.engine._activeReverseAjax && dwr.engine._pollReq) {
|
||
|
dwr.engine._pollReq.abort();
|
||
|
}
|
||
|
dwr.engine._activeReverseAjax = false;
|
||
|
}
|
||
|
// TODO: in iframe mode, if we start, stop, start then the second start may
|
||
|
// well kick off a second iframe while the first is still about to return
|
||
|
// we should cope with this but we don't
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* Turn server notification of page unload on and off
|
||
|
* @param {boolean} notify true or false depending on if we want to turn unload on or off
|
||
|
* @see getahead.org/dwr/browser/engine/options
|
||
|
*/
|
||
|
dwr.engine.setNotifyServerOnPageUnload = function(notify) {
|
||
|
dwr.engine._isNotifyServerOnPageUnload = notify;
|
||
|
};
|
||
|
|
||
|
/*
|
||
|
* The maximum number of retries before failure.
|
||
|
* @param - maxRetries - The maximum number of retries before failure.
|
||
|
*/
|
||
|
dwr.engine.setMaxRetries = function(maxRetries) {
|
||
|
dwr.engine._maxRetries = maxRetries;
|
||
|
};
|
||
|
|
||
|
/*
|
||
|
* The intervals between successive retries in seconds
|
||
|
* @param - array of integers representing the retry interval in seconds.
|
||
|
*/
|
||
|
dwr.engine.setRetryIntervals = function(intervalsArray) {
|
||
|
dwr.engine._retryIntervals = intervalsArray;
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* The default message handler.
|
||
|
* @param {String} message The text of the error message
|
||
|
* @param {Object} ex An error object containing at least a name and message
|
||
|
* @see getahead.org/dwr/browser/engine/errors
|
||
|
*/
|
||
|
dwr.engine.defaultErrorHandler = function(message, ex) {
|
||
|
dwr.engine._debug("Error: " + ex.name + ", " + ex.message, true);
|
||
|
if (message == null || message == "") alert("A server error has occurred.");
|
||
|
// Ignore NS_ERROR_NOT_AVAILABLE if Mozilla is being narky
|
||
|
//else if (message.indexOf("0x80040111") != -1) dwr.engine._debug(message);
|
||
|
//else alert(message);
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* The default warning handler.
|
||
|
* @param {String} message The text of the error message
|
||
|
* @param {Object} ex An error object containing at least a name and message
|
||
|
* @see getahead.org/dwr/browser/engine/errors
|
||
|
*/
|
||
|
dwr.engine.defaultWarningHandler = function(message, ex) {
|
||
|
dwr.engine._debug(message);
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* For reduced latency you can group several remote calls together using a batch.
|
||
|
* @see getahead.org/dwr/browser/engine/batch
|
||
|
*/
|
||
|
dwr.engine.beginBatch = function() {
|
||
|
if (dwr.engine._batch) {
|
||
|
dwr.engine._handleError(null, { name:"dwr.engine.batchBegun", message:"Batch already begun" });
|
||
|
return;
|
||
|
}
|
||
|
dwr.engine._batch = dwr.engine.batch.create();
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* Finished grouping a set of remote calls together. Go and execute them all.
|
||
|
* @param {Object} options A options object to customize processing
|
||
|
* @see getahead.org/dwr/browser/engine/batch
|
||
|
*/
|
||
|
dwr.engine.endBatch = function(options) {
|
||
|
var batch = dwr.engine._batch;
|
||
|
if (batch == null) {
|
||
|
dwr.engine._handleError(null, { name:"dwr.engine.batchNotBegun", message:"No batch in progress" });
|
||
|
return;
|
||
|
}
|
||
|
dwr.engine._batch = null;
|
||
|
if (batch.map.callCount == 0) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
// The hooks need to be merged carefully to preserve ordering
|
||
|
if (options) {
|
||
|
dwr.engine.batch.merge(batch, options);
|
||
|
}
|
||
|
|
||
|
// In ordered mode, we don't send unless the list of sent items is empty
|
||
|
if (dwr.engine._ordered && dwr.engine._batchesLength != 0) {
|
||
|
dwr.engine._batchQueue[dwr.engine._batchQueue.length] = batch;
|
||
|
}
|
||
|
else {
|
||
|
return dwr.engine.transport.send(batch);
|
||
|
}
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* For use with file downloads. When a DWR function returns a binary download
|
||
|
* you can prompt the user to save it using this function
|
||
|
* @param {Object} data The binary data passed from DWR
|
||
|
*/
|
||
|
dwr.engine.openInDownload = function(data) {
|
||
|
var div = document.createElement("div");
|
||
|
document.body.appendChild(div);
|
||
|
div.innerHTML = "<iframe width='0' height='0' scrolling='no' frameborder='0' src='" + data + "'></iframe>";
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* What is the current version DWR number
|
||
|
* DWR version numbers are of the form "Version 1.2.3.3128[.beta]", where:
|
||
|
* 1 is the major release number. Changes in major version number indicate
|
||
|
* significant enhancements in functionality
|
||
|
* 2 is the minor release number. Changes in minor version number indicate
|
||
|
* less significant changes in functionality
|
||
|
* 3 is the revision release number. Changes here typically indicate bug
|
||
|
* fixes only
|
||
|
* 3128 is the build number. This number increments for each build
|
||
|
* .beta is a release title that is generally only used for non production
|
||
|
* releases to indicate the purpose/quality of the release
|
||
|
* The label is these strings concatenated
|
||
|
*/
|
||
|
dwr.version = {
|
||
|
/**
|
||
|
* Changes in major version number indicate significant enhancements
|
||
|
*/
|
||
|
major:parseInt("3"),
|
||
|
|
||
|
/**
|
||
|
* Changes in minor version number indicate smaller enhancements
|
||
|
*/
|
||
|
minor:parseInt("0"),
|
||
|
|
||
|
/**
|
||
|
* Changes with the revision number typically indicate bug-fixes only
|
||
|
*/
|
||
|
revision:parseInt("0"),
|
||
|
|
||
|
/**
|
||
|
* The build number increments for each build
|
||
|
*/
|
||
|
build:parseInt("116"),
|
||
|
|
||
|
/**
|
||
|
* Only used for non production releases to indicate the purpose/quality of
|
||
|
* the release. Example titles include 'milestone1' or 'beta3'.
|
||
|
*/
|
||
|
title:"rc1",
|
||
|
|
||
|
/**
|
||
|
* The strings above concatenated
|
||
|
*/
|
||
|
label:"3.0.0.116.rc1"
|
||
|
};
|
||
|
|
||
|
//==============================================================================
|
||
|
// Only private stuff below here
|
||
|
//==============================================================================
|
||
|
|
||
|
/** The session cookie name */
|
||
|
dwr.engine._sessionCookieName = "JSESSIONID"; // JSESSIONID
|
||
|
|
||
|
/** Is GET enabled for the benefit of Safari? */
|
||
|
dwr.engine._allowGetForSafariButMakeForgeryEasier = "false";
|
||
|
|
||
|
/** The script prefix to strip in the case of scriptTagProtection. */
|
||
|
dwr.engine._scriptTagProtection = "throw 'allowScriptTagRemoting is false.';";
|
||
|
|
||
|
/** The default path to the DWR servlet */
|
||
|
dwr.engine._pathToDwrServlet = "/dwr";
|
||
|
|
||
|
/** Do we use XHR for reverse ajax because we are not streaming? */
|
||
|
dwr.engine._pollWithXhr = "false";
|
||
|
|
||
|
/** These URLs can be configured from the server */
|
||
|
dwr.engine._ModePlainCall = "/call/plaincall/";
|
||
|
dwr.engine._ModePlainPoll = "/call/plainpoll/";
|
||
|
dwr.engine._ModeHtmlCall = "/call/htmlcall/";
|
||
|
dwr.engine._ModeHtmlPoll = "/call/htmlpoll/";
|
||
|
|
||
|
/** Do we make the calls async? Default to 'true' */
|
||
|
dwr.engine._async = Boolean("true");
|
||
|
|
||
|
/** The page id */
|
||
|
dwr.engine._scriptSessionId = null;
|
||
|
|
||
|
/** A function to be called before requests are marshalled. Can be null. */
|
||
|
dwr.engine._preHook = null;
|
||
|
|
||
|
/** A function to be called after replies are received. Can be null. */
|
||
|
dwr.engine._postHook = null;
|
||
|
|
||
|
/** A map of the batches that we have sent and are awaiting a reply on. */
|
||
|
dwr.engine._batches = {};
|
||
|
|
||
|
/** A count of the number of outstanding batches. Should be == to _batches.length unless prototype has messed things up */
|
||
|
dwr.engine._batchesLength = 0;
|
||
|
|
||
|
/** In ordered mode, the array of batches waiting to be sent */
|
||
|
dwr.engine._batchQueue = [];
|
||
|
|
||
|
/** Do we attempt to ensure that calls happen in the order in which they were
|
||
|
sent? This starts true until we have fetched the ids, when it is to false */
|
||
|
dwr.engine._ordered = true;
|
||
|
|
||
|
/** The current batch (if we are in batch mode) */
|
||
|
dwr.engine._batch = null;
|
||
|
|
||
|
/** The global timeout */
|
||
|
dwr.engine._timeout = 0;
|
||
|
|
||
|
/** Are we doing comet or polling? */
|
||
|
dwr.engine._activeReverseAjax = false;
|
||
|
|
||
|
/** The xhr object that we are using to poll */
|
||
|
dwr.engine._pollReq = null;
|
||
|
|
||
|
/** How many milliseconds between internal comet polls */
|
||
|
dwr.engine._pollCometInterval = 200;
|
||
|
|
||
|
/** How many times have we re-tried a call? */
|
||
|
dwr.engine._retries = 0;
|
||
|
dwr.engine._maxRetries = 10;
|
||
|
|
||
|
/** The intervals between successive retries in seconds */
|
||
|
dwr.engine._retryIntervals = [];
|
||
|
/** Used as the default for reverse ajax/polling */
|
||
|
dwr.engine._defaultRetryIntervals = [ 2, 5, 10, 60, 300 ];
|
||
|
|
||
|
/** Do we do a document.reload if we get a text/html reply? */
|
||
|
dwr.engine._textHtmlHandler = null;
|
||
|
|
||
|
/** If you wish to send custom headers with every request */
|
||
|
dwr.engine._headers = null;
|
||
|
|
||
|
/** If you wish to send extra custom request attributes with each request */
|
||
|
dwr.engine._requestAttributes = null;
|
||
|
|
||
|
/** Batch ids allow us to know which batch the server is answering */
|
||
|
dwr.engine._nextBatchId = 0;
|
||
|
|
||
|
/** A list of the properties that need merging from calls to a batch */
|
||
|
dwr.engine._propnames = [ "async", "timeout", "errorHandler", "warningHandler", "textHtmlHandler" ];
|
||
|
|
||
|
/** Do we stream, or can be hacked to do so? */
|
||
|
dwr.engine._partialResponseNo = 0;
|
||
|
dwr.engine._partialResponseYes = 1;
|
||
|
dwr.engine._partialResponseFlush = 2;
|
||
|
|
||
|
/** Are we doing page unloading? */
|
||
|
dwr.engine._isNotifyServerOnPageUnload = false;
|
||
|
|
||
|
/**
|
||
|
* A map of all mapped classes whose class declarations have been loaded
|
||
|
* (dwrClassName -> constructor function)
|
||
|
* This could have been pre-created by interface scripts, so we need to check.
|
||
|
*/
|
||
|
if (typeof dwr.engine._mappedClasses == 'undefined') {
|
||
|
dwr.engine._mappedClasses = {};
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Find the HTTP session id sent by the web server
|
||
|
* @private
|
||
|
*/
|
||
|
dwr.engine._getHttpSessionId = function() {
|
||
|
// try to find the httpSessionId
|
||
|
var cookies = document.cookie.split(';');
|
||
|
for (var i = 0; i < cookies.length; i++) {
|
||
|
var cookie = cookies[i];
|
||
|
while (cookie.charAt(0) == ' ') cookie = cookie.substring(1, cookie.length);
|
||
|
if (cookie.indexOf(dwr.engine._sessionCookieName + "=") == 0) {
|
||
|
return cookie.substring(dwr.engine._sessionCookieName.length + 1, cookie.length);
|
||
|
}
|
||
|
}
|
||
|
return "";
|
||
|
};
|
||
|
|
||
|
/** A function to call if something fails. */
|
||
|
dwr.engine._errorHandler = dwr.engine.defaultErrorHandler;
|
||
|
|
||
|
/** For debugging when something unexplained happens. */
|
||
|
dwr.engine._warningHandler = dwr.engine.defaultWarningHandler;
|
||
|
|
||
|
/** Undocumented interceptors - do not use */
|
||
|
dwr.engine._postSeperator = "\n";
|
||
|
dwr.engine._defaultInterceptor = function(data) { return data; };
|
||
|
dwr.engine._urlRewriteHandler = dwr.engine._defaultInterceptor;
|
||
|
dwr.engine._contentRewriteHandler = dwr.engine._defaultInterceptor;
|
||
|
dwr.engine._replyRewriteHandler = dwr.engine._defaultInterceptor;
|
||
|
|
||
|
/** Is this page in the process of unloading? */
|
||
|
dwr.engine._unloading = false;
|
||
|
|
||
|
/** @private Abort any XHRs in progress at page unload (solves zombie socket problems in IE). */
|
||
|
dwr.engine._unloader = function() {
|
||
|
dwr.engine._unloading = true;
|
||
|
|
||
|
// Empty queue of waiting ordered requests
|
||
|
dwr.engine._batchQueue.length = 0;
|
||
|
|
||
|
// Abort any ongoing XHRs and clear their batches
|
||
|
var batch;
|
||
|
for (var batchId in dwr.engine._batches) {
|
||
|
batch = dwr.engine._batches[batchId];
|
||
|
// Only process objects that look like batches (avoid prototype additions!)
|
||
|
if (batch && batch.map) {
|
||
|
if (batch.req) {
|
||
|
batch.req.abort();
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// If we have used reverse ajax then we try to tell the server we are gone
|
||
|
if (dwr.engine._isNotifyServerOnPageUnload) {
|
||
|
dwr.engine._debug("calling unloader for: " + dwr.engine._scriptSessionId);
|
||
|
batch = {
|
||
|
map:{
|
||
|
callCount:1,
|
||
|
'c0-scriptName':'__System',
|
||
|
'c0-methodName':'pageUnloaded',
|
||
|
'c0-id':0
|
||
|
},
|
||
|
paramCount:0, isPoll:false, async:true,
|
||
|
headers:{}, preHooks:[], postHooks:[],
|
||
|
timeout:dwr.engine._timeout,
|
||
|
errorHandler:null, warningHandler:null, textHtmlHandler:null,
|
||
|
path:dwr.engine._pathToDwrServlet,
|
||
|
handlers:[{ exceptionHandler:null, callback:null }]
|
||
|
};
|
||
|
dwr.engine.transport.send(batch);
|
||
|
dwr.engine._isNotifyServerOnPageUnload = false;
|
||
|
}
|
||
|
};
|
||
|
|
||
|
// Now register the unload handler
|
||
|
if (!dwr.engine.isJaxerServer) {
|
||
|
if (window.addEventListener) window.addEventListener('unload', dwr.engine._unloader, false);
|
||
|
else if (window.attachEvent) window.attachEvent('onunload', dwr.engine._unloader);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Send a request. Called by the JavaScript interface stub
|
||
|
* @private
|
||
|
* @param path part of URL after the host and before the exec bit without leading or trailing /s
|
||
|
* @param scriptName The class to execute
|
||
|
* @param methodName The method on said class to execute
|
||
|
* @param func The callback function to which any returned data should be passed
|
||
|
* if this is null, any returned data will be ignored
|
||
|
* @param args The parameters to passed to the above method
|
||
|
*/
|
||
|
dwr.engine._execute = function(path, scriptName, methodName, args) {
|
||
|
var singleShot = false;
|
||
|
if (dwr.engine._batch == null) {
|
||
|
dwr.engine.beginBatch();
|
||
|
singleShot = true;
|
||
|
}
|
||
|
|
||
|
var batch = dwr.engine._batch;
|
||
|
// All the paths MUST be to the same servlet
|
||
|
if (batch.path == null) {
|
||
|
batch.path = path;
|
||
|
}
|
||
|
else {
|
||
|
if (batch.path != path) {
|
||
|
dwr.engine._handleError(batch, { name:"dwr.engine.multipleServlets", message:"Can't batch requests to multiple DWR Servlets." });
|
||
|
return;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
dwr.engine.batch.addCall(batch, scriptName, methodName, args);
|
||
|
|
||
|
// Now we have finished remembering the call, we increment the call count
|
||
|
batch.map.callCount++;
|
||
|
if (singleShot) {
|
||
|
return dwr.engine.endBatch();
|
||
|
}
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* Poll the server to see if there is any data waiting
|
||
|
* @private
|
||
|
*/
|
||
|
dwr.engine._poll = function() {
|
||
|
if (!dwr.engine._activeReverseAjax) {
|
||
|
return;
|
||
|
}
|
||
|
var batch = dwr.engine.batch.createPoll();
|
||
|
dwr.engine.transport.send(batch);
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* Try to recover from errors
|
||
|
* @param {Object} msg
|
||
|
* @param {Object} ex
|
||
|
*/
|
||
|
dwr.engine._retryHandler = function(batch, ex) {
|
||
|
if (dwr.engine._retries >= dwr.engine._maxRetries) {
|
||
|
dwr.engine._activeReverseAjax = false;
|
||
|
if (batch && typeof batch.errorHandler == "function") {
|
||
|
batch.errorHandler(ex.message, ex);
|
||
|
} else if (dwr.engine._errorHandler) {
|
||
|
dwr.engine._errorHandler(ex.message, ex);
|
||
|
}
|
||
|
if (batch) dwr.engine.batch.remove(batch);
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
var index = dwr.engine._retries;
|
||
|
if (index >= dwr.engine._retryIntervals.length) {
|
||
|
index = dwr.engine._retryIntervals.length - 1;
|
||
|
}
|
||
|
|
||
|
dwr.engine._debug("Reverse Ajax poll failed (retries=" + dwr.engine._retries + "). Trying again in " + dwr.engine._retryIntervals[index] + "s: " + ex.name + " : " + ex.message);
|
||
|
setTimeout(dwr.engine._poll, 1000 * dwr.engine._retryIntervals[index]);
|
||
|
|
||
|
dwr.engine._retries++;
|
||
|
};
|
||
|
|
||
|
/** @private This is a hack to make the context be this window */
|
||
|
dwr.engine._eval = function(script) {
|
||
|
if (script == null) {
|
||
|
return null;
|
||
|
}
|
||
|
if (script == "") {
|
||
|
dwr.engine._debug("Warning: blank script", true);
|
||
|
return null;
|
||
|
}
|
||
|
// dwr.engine._debug("Exec: [" + script + "]", true);
|
||
|
return eval(script);
|
||
|
};
|
||
|
|
||
|
/** @private call all the post hooks for a batch */
|
||
|
dwr.engine._callPostHooks = function(batch) {
|
||
|
if (batch.postHooks) {
|
||
|
for (var i = 0; i < batch.postHooks.length; i++) {
|
||
|
batch.postHooks[i]();
|
||
|
}
|
||
|
batch.postHooks = null;
|
||
|
}
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* Generic error handling routing to save having null checks everywhere
|
||
|
* @private
|
||
|
* @param {Object} batch
|
||
|
* @param {Object} ex
|
||
|
*/
|
||
|
dwr.engine._handleError = function(batch, ex) {
|
||
|
dwr.engine._prepareException(ex);
|
||
|
// If a retry policy has been specified, call the retryHandler.
|
||
|
if (dwr.engine._retryIntervals.length > 0) {
|
||
|
dwr.engine._retryHandler(batch, ex);
|
||
|
} else {
|
||
|
if (batch && typeof batch.errorHandler == "function") batch.errorHandler(ex.message, ex);
|
||
|
else if (dwr.engine._errorHandler) dwr.engine._errorHandler(ex.message, ex);
|
||
|
if (batch) dwr.engine.batch.remove(batch);
|
||
|
}
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* Generic error handling routing to save having null checks everywhere
|
||
|
* @private
|
||
|
* @param {Object} batch
|
||
|
* @param {Object} ex
|
||
|
*/
|
||
|
dwr.engine._handleWarning = function(batch, ex) {
|
||
|
dwr.engine._prepareException(ex);
|
||
|
// If a retry policy has been specified, call the retryHandler.
|
||
|
if (dwr.engine._retryIntervals.length > 0) {
|
||
|
return dwr.engine._retryHandler(batch, ex);
|
||
|
} else {
|
||
|
if (batch && typeof batch.warningHandler == "function") batch.warningHandler(ex.message, ex);
|
||
|
else if (dwr.engine._warningHandler) dwr.engine._warningHandler(ex.message, ex);
|
||
|
if (batch) dwr.engine.batch.remove(batch);
|
||
|
}
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* Prepares an exception for an error/warning handler.
|
||
|
* @private
|
||
|
* @param {Object} ex
|
||
|
*/
|
||
|
dwr.engine._prepareException = function(ex) {
|
||
|
if (typeof ex == "string") ex = { name:"unknown", message:ex };
|
||
|
if (ex.message == null) ex.message = "";
|
||
|
if (ex.name == null) ex.name = "unknown";
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* Used internally when some message needs to get to the programmer
|
||
|
* @private
|
||
|
* @param {String} message
|
||
|
* @param {Object} stacktrace
|
||
|
*/
|
||
|
dwr.engine._debug = function(message, stacktrace) {
|
||
|
var written = false;
|
||
|
try {
|
||
|
if (window.console) {
|
||
|
if (stacktrace && window.console.trace) window.console.trace();
|
||
|
window.console.log(message);
|
||
|
written = true;
|
||
|
}
|
||
|
else if (window.opera && window.opera.postError) {
|
||
|
window.opera.postError(message);
|
||
|
written = true;
|
||
|
}
|
||
|
else if (window.Jaxer && Jaxer.isOnServer) {
|
||
|
Jaxer.Log.info(message);
|
||
|
written = true;
|
||
|
}
|
||
|
}
|
||
|
catch (ex) { /* ignore */ }
|
||
|
|
||
|
if (!written) {
|
||
|
var debug = document.getElementById("dwr-debug");
|
||
|
if (debug) {
|
||
|
var contents = message + "<br/>" + debug.innerHTML;
|
||
|
if (contents.length > 2048) contents = contents.substring(0, 2048);
|
||
|
debug.innerHTML = contents;
|
||
|
}
|
||
|
}
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* Functions called by the server
|
||
|
*/
|
||
|
dwr.engine.remote = {
|
||
|
/**
|
||
|
* Execute a callback
|
||
|
* @private
|
||
|
* @param {int} batchId The ID of the batch that we are replying to
|
||
|
* @param {int} callId The call ID that the script relates to
|
||
|
* @param {String} reply The script to execute
|
||
|
*/
|
||
|
handleCallback:function(batchId, callId, reply) {
|
||
|
var batch = dwr.engine._batches[batchId];
|
||
|
if (batch == null) {
|
||
|
dwr.engine._debug("Warning: batch == null in remoteHandleCallback for batchId=" + batchId, true);
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
// We store the reply in the batch so that in sync mode we can return the data
|
||
|
batch.reply = reply;
|
||
|
|
||
|
// Error handlers inside here indicate an error that is nothing to do
|
||
|
// with DWR so we handle them differently.
|
||
|
try {
|
||
|
var handlers = batch.handlers[callId];
|
||
|
if (!handlers) {
|
||
|
dwr.engine._debug("Warning: Missing handlers. callId=" + callId, true);
|
||
|
}
|
||
|
else {
|
||
|
batch.handlers[callId].completed = true;
|
||
|
if (typeof handlers.callback == "function") {
|
||
|
handlers.callback.apply(handlers.callbackScope, [ reply, handlers.callbackArg ]);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
catch (ex) {
|
||
|
dwr.engine._handleError(batch, ex);
|
||
|
}
|
||
|
},
|
||
|
|
||
|
/**
|
||
|
* Called by the server when a JavascriptFunction is executed
|
||
|
* @param id The ID of the serialized function
|
||
|
* @param args The arguments to pass to the function
|
||
|
*/
|
||
|
handleFunctionCall:function(id, args) {
|
||
|
var func = dwr.engine.serialize.remoteFunctions[id];
|
||
|
func.apply(window, args);
|
||
|
},
|
||
|
|
||
|
/**
|
||
|
* Called by the server when a JavascriptFunction is executed
|
||
|
* @param id The ID of the serialized function
|
||
|
* @param args The arguments to pass to the function
|
||
|
*/
|
||
|
handleObjectCall:function(id, methodName, args) {
|
||
|
var obj = dwr.engine.serialize.remoteFunctions[id];
|
||
|
obj[methodName].apply(obj, args);
|
||
|
},
|
||
|
|
||
|
/**
|
||
|
* Called by the server when a JavascriptFunction is executed
|
||
|
* @param propertyName The ID of the serialized function
|
||
|
* @param data The arguments to pass to the function
|
||
|
*/
|
||
|
handleSetCall:function(id, propertyName, data) {
|
||
|
var obj = dwr.engine.serialize.remoteFunctions[id];
|
||
|
obj[propertyName] = data;
|
||
|
},
|
||
|
|
||
|
/**
|
||
|
* Called by the server when a JavascriptFunction is closed
|
||
|
* @param id The ID of the serialized function
|
||
|
*/
|
||
|
handleFunctionClose:function(id) {
|
||
|
delete dwr.engine.serialize.remoteFunctions[id];
|
||
|
},
|
||
|
|
||
|
/**
|
||
|
* Called by the server: Handle an exception for a call
|
||
|
* @private
|
||
|
* @param {int} batchId The ID of the batch that we are replying to
|
||
|
* @param {int} callId The call ID that the script relates to
|
||
|
* @param {String} reply The script to execute
|
||
|
*/
|
||
|
handleException:function(batchId, callId, ex) {
|
||
|
var batch = dwr.engine._batches[batchId];
|
||
|
if (batch == null) {
|
||
|
dwr.engine._debug("Warning: null batch in remoteHandleException", true);
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
var handlers = batch.handlers[callId];
|
||
|
batch.handlers[callId].completed = true;
|
||
|
if (handlers == null) {
|
||
|
dwr.engine._debug("Warning: null handlers in remoteHandleException", true);
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
if (ex.message == undefined) {
|
||
|
ex.message = "";
|
||
|
}
|
||
|
|
||
|
if (typeof handlers.exceptionHandler == "function") {
|
||
|
handlers.exceptionHandler.call(handlers.exceptionScope, ex.message, ex, handlers.exceptionArg);
|
||
|
}
|
||
|
else if (typeof batch.errorHandler == "function") {
|
||
|
batch.errorHandler(ex.message, ex);
|
||
|
}
|
||
|
},
|
||
|
|
||
|
/**
|
||
|
* Called by the server: The whole batch is broken
|
||
|
* @private
|
||
|
* @param {Object} ex The data about what broke
|
||
|
* @param {int} batchId The ID of the batch that we are replying to
|
||
|
*/
|
||
|
handleBatchException:function(ex, batchId) {
|
||
|
var searchBatch = (dwr.engine._receivedBatch == null && batchId != null);
|
||
|
if (searchBatch) {
|
||
|
dwr.engine._receivedBatch = dwr.engine._batches[batchId];
|
||
|
}
|
||
|
if (ex.message == undefined) ex.message = "";
|
||
|
dwr.engine._handleError(dwr.engine._receivedBatch, ex);
|
||
|
if (searchBatch) {
|
||
|
dwr.engine._receivedBatch = null;
|
||
|
dwr.engine.batch.remove(dwr.engine._batches[batchId]);
|
||
|
}
|
||
|
},
|
||
|
|
||
|
/**
|
||
|
* Called by the server when we need to set a new script session id
|
||
|
* @param {Object} newSessionId The new script session id to be used from now
|
||
|
*/
|
||
|
handleNewScriptSession:function(newSessionId) {
|
||
|
if (dwr.engine._scriptSessionId != null) {
|
||
|
dwr.engine._debug("Server side script session id timed out. New session automatically created");
|
||
|
}
|
||
|
dwr.engine._scriptSessionId = newSessionId;
|
||
|
},
|
||
|
|
||
|
/**
|
||
|
* Called by the server when we need to set a new script session id
|
||
|
* @param {Object} newSessionId The new script session id to be used from now
|
||
|
*/
|
||
|
handleNewWindowName:function(windowName) {
|
||
|
dwr.engine._debug("Setting new window name: " + windowName);
|
||
|
if (window.name != null && window.name != "") {
|
||
|
dwr.engine._debug("- Warning: This will override existing name of: " + window.name);
|
||
|
}
|
||
|
window.name = windowName;
|
||
|
},
|
||
|
|
||
|
/**
|
||
|
* Execute some script in a different window
|
||
|
* @param {Object} windowName The name of the window in which to eval the script
|
||
|
* @param {Object} script The script to eval elsewhere
|
||
|
*/
|
||
|
handleForeign:function(windowName, script) {
|
||
|
var foreign = window.open(null, windowName);
|
||
|
if (foreign != null) {
|
||
|
if (foreign.dwr != null) {
|
||
|
foreign.dwr.engine._eval(script);
|
||
|
}
|
||
|
else {
|
||
|
dwr.engine._debug("Found window, but DWR did not exist in it");
|
||
|
}
|
||
|
}
|
||
|
else {
|
||
|
dwr.engine._debug("Could not find window");
|
||
|
}
|
||
|
},
|
||
|
|
||
|
/**
|
||
|
* Called by the server: Reverse ajax should not be used
|
||
|
* @private
|
||
|
* @param {Object} ex
|
||
|
* @param {int} batchId
|
||
|
*/
|
||
|
pollCometDisabled:function(ex, batchId){
|
||
|
dwr.engine.setActiveReverseAjax(false);
|
||
|
var searchBatch = (dwr.engine._receivedBatch == null && batchId != null);
|
||
|
if (searchBatch) {
|
||
|
dwr.engine._receivedBatch = dwr.engine._batches[batchId];
|
||
|
}
|
||
|
if (ex.message == undefined) {
|
||
|
ex.message = "";
|
||
|
}
|
||
|
dwr.engine._handleError(dwr.engine._receivedBatch, ex);
|
||
|
if (searchBatch) {
|
||
|
dwr.engine._receivedBatch = null;
|
||
|
dwr.engine.batch.remove(dwr.engine._batches[batchId]);
|
||
|
}
|
||
|
},
|
||
|
|
||
|
/**
|
||
|
* Called by the server: Create a new object of a mapped class
|
||
|
* @private
|
||
|
* @param {string} dwrClassName the name of the mapped class
|
||
|
* @param {Object} memberMap the object's data members
|
||
|
*/
|
||
|
newObject:function(dwrClassName, memberMap){
|
||
|
var classfunc = dwr.engine._mappedClasses[dwrClassName];
|
||
|
if (classfunc && classfunc.createFromMap) {
|
||
|
return classfunc.createFromMap(memberMap);
|
||
|
}
|
||
|
else {
|
||
|
memberMap.$dwrClassName = dwrClassName;
|
||
|
return memberMap;
|
||
|
}
|
||
|
}
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* Functions to serialize a data set into a list of parameters
|
||
|
*/
|
||
|
dwr.engine.serialize = {
|
||
|
/**
|
||
|
* ActiveX objects to use when we want to convert an xml string into a DOM object
|
||
|
*/
|
||
|
domDocument:[
|
||
|
"Msxml2.DOMDocument.6.0",
|
||
|
"Msxml2.DOMDocument.5.0",
|
||
|
"Msxml2.DOMDocument.4.0",
|
||
|
"Msxml2.DOMDocument.3.0",
|
||
|
"MSXML2.DOMDocument",
|
||
|
"MSXML.DOMDocument",
|
||
|
"Microsoft.XMLDOM"
|
||
|
],
|
||
|
|
||
|
/**
|
||
|
* A holder for functions that we have serialized for remote calling.
|
||
|
*/
|
||
|
remoteFunctions:{},
|
||
|
|
||
|
/**
|
||
|
* The ID of the next function that we serialize
|
||
|
*/
|
||
|
funcId:0,
|
||
|
|
||
|
/**
|
||
|
* Convert a text representation of XML into a DOM element
|
||
|
* @param {String} xml An xml string
|
||
|
*/
|
||
|
toDomElement:function(xml) {
|
||
|
return dwr.engine.serialize.toDomDocument(xml).documentElement;
|
||
|
},
|
||
|
|
||
|
/**
|
||
|
* Convert a text representation of XML into a DOM document
|
||
|
* @param {String} xml An xml string
|
||
|
*/
|
||
|
toDomDocument:function(xml) {
|
||
|
var dom;
|
||
|
if (window.DOMParser) {
|
||
|
var parser = new DOMParser();
|
||
|
dom = parser.parseFromString(xml, "text/xml");
|
||
|
if (!dom.documentElement || dom.documentElement.tagName == "parsererror") {
|
||
|
var message = dom.documentElement.firstChild.data;
|
||
|
message += "\n" + dom.documentElement.firstChild.nextSibling.firstChild.data;
|
||
|
throw message;
|
||
|
}
|
||
|
return dom;
|
||
|
}
|
||
|
else if (window.ActiveXObject) {
|
||
|
dom = dwr.engine.util.newActiveXObject(dwr.engine.serialize.domDocument);
|
||
|
dom.loadXML(xml); // What happens on parse fail with IE?
|
||
|
return dom;
|
||
|
}
|
||
|
else {
|
||
|
var div = document.createElement("div");
|
||
|
div.innerHTML = xml;
|
||
|
return div;
|
||
|
}
|
||
|
},
|
||
|
|
||
|
/**
|
||
|
* Marshall a data item
|
||
|
* @private
|
||
|
* @param batch A map of variables to how they have been marshalled
|
||
|
* @param referto An array of already marshalled variables to prevent recurrsion
|
||
|
* @param data The data to be marshalled
|
||
|
* @param name The name of the data being marshalled
|
||
|
*/
|
||
|
convert:function(batch, referto, data, name, depth) {
|
||
|
if (data == null) {
|
||
|
batch.map[name] = "null:null";
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
switch (typeof data) {
|
||
|
case "boolean":
|
||
|
batch.map[name] = "boolean:" + data;
|
||
|
break;
|
||
|
case "number":
|
||
|
batch.map[name] = "number:" + data;
|
||
|
break;
|
||
|
case "string":
|
||
|
batch.map[name] = "string:" + encodeURIComponent(data);
|
||
|
break;
|
||
|
case "object":
|
||
|
var ref = dwr.engine.serialize.lookup(referto, data, name);
|
||
|
var objstr = Object.prototype.toString.call(data);
|
||
|
if (data.$dwrByRef) batch.map[name] = dwr.engine.serialize.convertByReference(batch, referto, data, name, depth + 1);
|
||
|
else if (ref != null) batch.map[name] = ref;
|
||
|
else if (objstr == "[object String]") batch.map[name] = "string:" + encodeURIComponent(data);
|
||
|
else if (objstr == "[object Boolean]") batch.map[name] = "boolean:" + data;
|
||
|
else if (objstr == "[object Number]") batch.map[name] = "number:" + data;
|
||
|
else if (objstr == "[object Date]") batch.map[name] = "date:" + data.getTime();
|
||
|
else if (objstr == "[object Array]") batch.map[name] = dwr.engine.serialize.convertArray(batch, referto, data, name, depth + 1);
|
||
|
else if (data && data.tagName && data.tagName.toLowerCase() == "input" && data.type && data.type.toLowerCase() == "file") {
|
||
|
batch.fileUpload = true;
|
||
|
batch.map[name] = data;
|
||
|
}
|
||
|
else {
|
||
|
// This check for an HTML is not complete, but is there a better way?
|
||
|
// Maybe we should add: data.hasChildNodes typeof "function" == true
|
||
|
if (data.nodeName && data.nodeType) {
|
||
|
batch.map[name] = dwr.engine.serialize.convertXml(batch, referto, data, name, depth + 1);
|
||
|
}
|
||
|
else {
|
||
|
batch.map[name] = dwr.engine.serialize.convertObject(batch, referto, data, name, depth + 1);
|
||
|
}
|
||
|
}
|
||
|
break;
|
||
|
case "function":
|
||
|
// Ignore functions unless they are directly passed in
|
||
|
if (depth == 0) {
|
||
|
batch.map[name] = dwr.engine.serialize.convertByReference(batch, referto, data, name, depth + 1);
|
||
|
}
|
||
|
break;
|
||
|
default:
|
||
|
dwr.engine._handleWarning(null, { name:"dwr.engine.unexpectedType", message:"Unexpected type: " + typeof data + ", attempting default converter." });
|
||
|
batch.map[name] = "default:" + data;
|
||
|
break;
|
||
|
}
|
||
|
},
|
||
|
|
||
|
/**
|
||
|
* Marshall an object by reference
|
||
|
* @private
|
||
|
* @see dwr.engine.serialize.convert() for parameter details
|
||
|
*/
|
||
|
convertByReference:function(batch, referto, data, name, depth) {
|
||
|
var funcId = "f" + dwr.engine.serialize.funcId;
|
||
|
dwr.engine.serialize.remoteFunctions[funcId] = data;
|
||
|
dwr.engine.serialize.funcId++;
|
||
|
return "byref:" + funcId;
|
||
|
},
|
||
|
|
||
|
/**
|
||
|
* Marshall an array
|
||
|
* @private
|
||
|
* @see dwr.engine.serialize.convert() for parameter details
|
||
|
*/
|
||
|
convertArray:function(batch, referto, data, name, depth) {
|
||
|
var childName, i;
|
||
|
if (dwr.engine.isIE <= 7) {
|
||
|
// Use array joining on IE1-7 (fastest)
|
||
|
var buf = ["array:["];
|
||
|
for (i = 0; i < data.length; i++) {
|
||
|
if (i != 0) buf.push(",");
|
||
|
batch.paramCount++;
|
||
|
childName = "c" + dwr.engine._batch.map.callCount + "-e" + batch.paramCount;
|
||
|
dwr.engine.serialize.convert(batch, referto, data[i], childName, depth + 1);
|
||
|
buf.push("reference:");
|
||
|
buf.push(childName);
|
||
|
}
|
||
|
buf.push("]");
|
||
|
reply = buf.join("");
|
||
|
}
|
||
|
else {
|
||
|
// Use string concat on other browsers (fastest)
|
||
|
var reply = "array:[";
|
||
|
for (i = 0; i < data.length; i++) {
|
||
|
if (i != 0) reply += ",";
|
||
|
batch.paramCount++;
|
||
|
childName = "c" + dwr.engine._batch.map.callCount + "-e" + batch.paramCount;
|
||
|
dwr.engine.serialize.convert(batch, referto, data[i], childName, depth + 1);
|
||
|
reply += "reference:";
|
||
|
reply += childName;
|
||
|
}
|
||
|
reply += "]";
|
||
|
}
|
||
|
|
||
|
return reply;
|
||
|
},
|
||
|
|
||
|
/**
|
||
|
* Marshall an object by value
|
||
|
* @private
|
||
|
* @see dwr.engine.serialize.convert() for parameter details
|
||
|
*/
|
||
|
convertObject:function(batch, referto, data, name, depth) {
|
||
|
// treat objects as an associative arrays
|
||
|
var reply = "Object_" + dwr.engine.serialize.getObjectClassName(data) + ":{";
|
||
|
var elementset = (data.constructor && data.constructor.$dwrClassMembers ? data.constructor.$dwrClassMembers : data);
|
||
|
var element;
|
||
|
for (element in elementset) {
|
||
|
if (typeof data[element] != "function") {
|
||
|
batch.paramCount++;
|
||
|
var childName = "c" + dwr.engine._batch.map.callCount + "-e" + batch.paramCount;
|
||
|
dwr.engine.serialize.convert(batch, referto, data[element], childName, depth + 1);
|
||
|
reply += encodeURIComponent(element) + ":reference:" + childName + ", ";
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (reply.substring(reply.length - 2) == ", ") {
|
||
|
reply = reply.substring(0, reply.length - 2);
|
||
|
}
|
||
|
reply += "}";
|
||
|
|
||
|
return reply;
|
||
|
},
|
||
|
|
||
|
/**
|
||
|
* Marshall an object
|
||
|
* @private
|
||
|
* @see dwr.engine.serialize.convert() for parameter details
|
||
|
*/
|
||
|
convertXml:function(batch, referto, data, name, depth) {
|
||
|
var output;
|
||
|
if (window.XMLSerializer) output = new XMLSerializer().serializeToString(data);
|
||
|
else if (data.toXml) output = data.toXml;
|
||
|
else output = data.innerHTML;
|
||
|
|
||
|
return "xml:" + encodeURIComponent(output);
|
||
|
},
|
||
|
|
||
|
/**
|
||
|
* Have we already converted this object?
|
||
|
* @private
|
||
|
* @see dwr.engine.serialize.convert() for parameter details
|
||
|
*/
|
||
|
lookup:function(referto, data, name) {
|
||
|
var lookup;
|
||
|
// Can't use a map: getahead.org/ajax/javascript-gotchas
|
||
|
for (var i = 0; i < referto.length; i++) {
|
||
|
if (referto[i].data == data) {
|
||
|
lookup = referto[i];
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
if (lookup) {
|
||
|
return "reference:" + lookup.name;
|
||
|
}
|
||
|
referto.push({ data:data, name:name });
|
||
|
return null;
|
||
|
},
|
||
|
|
||
|
/**
|
||
|
* Returns the classname of supplied argument obj. Similar to typeof, but
|
||
|
* which returns the name of the constructor that created the object rather
|
||
|
* than 'object'
|
||
|
* @private
|
||
|
* @param {Object} obj The object to detect the type of
|
||
|
* @return The name of the object
|
||
|
*/
|
||
|
getObjectClassName:function(obj) {
|
||
|
// Different handling depending on if, and what type of, class-mapping is used
|
||
|
if (obj.$dwrClassName)
|
||
|
return obj.$dwrClassName; // Light class-mapping uses the classname from a property on the instance
|
||
|
else if (obj.constructor && obj.constructor.$dwrClassName)
|
||
|
return obj.constructor.$dwrClassName; // Full class-mapping uses the classname from a property on the constructor function
|
||
|
else
|
||
|
return "Object";
|
||
|
}
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* Functions to handle the various remoting transport
|
||
|
*/
|
||
|
dwr.engine.transport = {
|
||
|
/**
|
||
|
* Actually send the block of data in the batch object.
|
||
|
* @private
|
||
|
* @param {Object} batch
|
||
|
*/
|
||
|
send:function(batch) {
|
||
|
dwr.engine.batch.prepareToSend(batch);
|
||
|
|
||
|
// Work out if we are going cross domain
|
||
|
var isCrossDomain = false;
|
||
|
if (batch.path == null) {
|
||
|
batch.path = dwr.engine._pathToDwrServlet;
|
||
|
}
|
||
|
if (batch.path.indexOf("http://") == 0 || batch.path.indexOf("https://") == 0) {
|
||
|
var dwrShortPath = dwr.engine._pathToDwrServlet.split("/", 3).join("/");
|
||
|
var hrefShortPath = window.location.href.split("/", 3).join("/");
|
||
|
isCrossDomain = (dwrShortPath != hrefShortPath);
|
||
|
}
|
||
|
|
||
|
if (batch.fileUpload) {
|
||
|
if (isCrossDomain) {
|
||
|
throw new Error("Cross domain file uploads are not possible with this release of DWR");
|
||
|
}
|
||
|
batch.transport = dwr.engine.transport.iframe;
|
||
|
}
|
||
|
else if (isCrossDomain && !dwr.engine.isJaxerServer) {
|
||
|
batch.transport = dwr.engine.transport.scriptTag;
|
||
|
}
|
||
|
// else if (batch.isPoll && dwr.engine.isIE) {
|
||
|
// batch.transport = dwr.engine.transport.htmlfile;
|
||
|
// }
|
||
|
else {
|
||
|
batch.transport = dwr.engine.transport.xhr;
|
||
|
}
|
||
|
|
||
|
return batch.transport.send(batch);
|
||
|
},
|
||
|
|
||
|
/**
|
||
|
* A generic function to remove all remoting artifacts
|
||
|
* @param {Object} batch The batch that has completed
|
||
|
*/
|
||
|
remove:function(batch) {
|
||
|
dwr.engine.transport.iframe.remove(batch);
|
||
|
dwr.engine.transport.xhr.remove(batch);
|
||
|
},
|
||
|
|
||
|
/**
|
||
|
* Called as a result of a request timeout
|
||
|
* @private
|
||
|
* @param {Object} batch The batch that is aborting
|
||
|
*/
|
||
|
abort:function(batch) {
|
||
|
if (batch && !batch.completed) {
|
||
|
dwr.engine.batch.remove(batch);
|
||
|
|
||
|
if (batch.req) {
|
||
|
batch.req.abort();
|
||
|
}
|
||
|
|
||
|
dwr.engine.transport.remove(batch);
|
||
|
dwr.engine._handleError(batch, { name:"dwr.engine.timeout", message:"Timeout" });
|
||
|
}
|
||
|
},
|
||
|
|
||
|
/**
|
||
|
* Remoting through XHR
|
||
|
*/
|
||
|
xhr:{
|
||
|
/**
|
||
|
* The default HTTP method to use
|
||
|
*/
|
||
|
httpMethod:"POST",
|
||
|
|
||
|
/**
|
||
|
* The ActiveX objects to use when we want to do an XMLHttpRequest call.
|
||
|
* TODO: We arrived at this by trial and error. Other toolkits use
|
||
|
* different strings, maybe there is an officially correct version?
|
||
|
*/
|
||
|
XMLHTTP:["Msxml2.XMLHTTP.6.0", "Msxml2.XMLHTTP.5.0", "Msxml2.XMLHTTP.4.0", "MSXML2.XMLHTTP.3.0", "MSXML2.XMLHTTP", "Microsoft.XMLHTTP"],
|
||
|
|
||
|
/**
|
||
|
* Setup a batch for transfer through XHR
|
||
|
* @param {Object} batch The batch to alter for XHR transmit
|
||
|
*/
|
||
|
send:function(batch) {
|
||
|
if (batch.isPoll) {
|
||
|
batch.map.partialResponse = dwr.engine._partialResponseYes;
|
||
|
}
|
||
|
|
||
|
// Do proxies or IE force us to use early closing mode?
|
||
|
if (batch.isPoll && dwr.engine._pollWithXhr == "true") {
|
||
|
batch.map.partialResponse = dwr.engine._partialResponseNo;
|
||
|
}
|
||
|
if (batch.isPoll && dwr.engine.isIE < 8) {
|
||
|
batch.map.partialResponse = dwr.engine._partialResponseNo;
|
||
|
} else if (batch.isPoll && dwr.engine.isIE >= 8) {
|
||
|
batch.map.partialResponse = dwr.engine._partialResponseFlush;
|
||
|
}
|
||
|
|
||
|
if (window.XMLHttpRequest) {
|
||
|
batch.req = new XMLHttpRequest();
|
||
|
}
|
||
|
else if (window.ActiveXObject) {
|
||
|
batch.req = dwr.engine.util.newActiveXObject(dwr.engine.transport.xhr.XMLHTTP);
|
||
|
}
|
||
|
|
||
|
// Proceed using XMLHttpRequest
|
||
|
if (batch.async == true) {
|
||
|
batch.req.onreadystatechange = function() {
|
||
|
if (typeof dwr != 'undefined') {
|
||
|
dwr.engine.transport.xhr.stateChange(batch);
|
||
|
}
|
||
|
};
|
||
|
}
|
||
|
|
||
|
// If we're polling, record this for monitoring
|
||
|
if (batch.isPoll) {
|
||
|
dwr.engine._pollReq = batch.req;
|
||
|
// In IE XHR is an ActiveX control so you can't augment it like this
|
||
|
if (!dwr.engine.isIE) batch.req.batch = batch;
|
||
|
}
|
||
|
|
||
|
httpMethod = dwr.engine.transport.xhr.httpMethod;
|
||
|
|
||
|
// Workaround for Safari 1.x POST bug
|
||
|
var indexSafari = navigator.userAgent.indexOf("Safari/");
|
||
|
if (indexSafari >= 0) {
|
||
|
var version = navigator.userAgent.substring(indexSafari + 7);
|
||
|
if (parseInt(version, 10) < 400) {
|
||
|
if (dwr.engine._allowGetForSafariButMakeForgeryEasier == "true") {
|
||
|
httpMethod = "GET";
|
||
|
}
|
||
|
else {
|
||
|
dwr.engine._handleWarning(batch, {
|
||
|
name: "dwr.engine.oldSafari",
|
||
|
message: "Safari GET support disabled. See getahead.org/dwr/server/servlet and allowGetForSafariButMakeForgeryEasier."
|
||
|
});
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
batch.mode = batch.isPoll ? dwr.engine._ModePlainPoll : dwr.engine._ModePlainCall;
|
||
|
var request = dwr.engine.batch.constructRequest(batch, httpMethod);
|
||
|
|
||
|
try {
|
||
|
batch.req.open(httpMethod, request.url, batch.async);
|
||
|
try {
|
||
|
for (var prop in batch.headers) {
|
||
|
var value = batch.headers[prop];
|
||
|
if (typeof value == "string") {
|
||
|
batch.req.setRequestHeader(prop, value);
|
||
|
}
|
||
|
}
|
||
|
if (!batch.headers["Content-Type"]) {
|
||
|
batch.req.setRequestHeader("Content-Type", "text/plain");
|
||
|
}
|
||
|
}
|
||
|
catch (ex) {
|
||
|
dwr.engine._handleWarning(batch, ex);
|
||
|
}
|
||
|
batch.req.send(request.body);
|
||
|
if (batch.async == false) {
|
||
|
dwr.engine.transport.xhr.stateChange(batch);
|
||
|
}
|
||
|
}
|
||
|
catch (ex) {
|
||
|
dwr.engine._handleError(batch, ex);
|
||
|
}
|
||
|
|
||
|
if (batch.isPoll && batch.map.partialResponse == dwr.engine._partialResponseYes) {
|
||
|
dwr.engine.transport.xhr.checkCometPoll();
|
||
|
}
|
||
|
|
||
|
// This is only of any use in sync mode to return the reply data
|
||
|
return batch.reply;
|
||
|
},
|
||
|
|
||
|
/**
|
||
|
* Called by XMLHttpRequest to indicate that something has happened
|
||
|
* @private
|
||
|
* @param {Object} batch The current remote operation
|
||
|
*/
|
||
|
stateChange:function(batch) {
|
||
|
var toEval;
|
||
|
|
||
|
if (batch.completed) {
|
||
|
dwr.engine._debug("Error: _stateChange() with batch.completed");
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
var req = batch.req;
|
||
|
try {
|
||
|
var readyState = req.readyState;
|
||
|
var notReady = (req.readyState != 4);
|
||
|
if (notReady) {
|
||
|
return;
|
||
|
}
|
||
|
}
|
||
|
catch (ex) {
|
||
|
dwr.engine._handleWarning(batch, ex);
|
||
|
// It's broken - clear up and forget this call
|
||
|
dwr.engine.batch.remove(batch);
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
if (dwr.engine._unloading && !dwr.engine.isJaxerServer) {
|
||
|
dwr.engine._debug("Ignoring reply from server as page is unloading.");
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
try {
|
||
|
var reply = req.responseText;
|
||
|
reply = dwr.engine._replyRewriteHandler(reply);
|
||
|
var status = req.status; // causes Mozilla to except on page moves
|
||
|
|
||
|
if (reply == null || reply == "") {
|
||
|
|
||
|
}
|
||
|
else if (status != 200) {
|
||
|
dwr.engine._handleError(batch, { name:"dwr.engine.http." + status, message:req.statusText });
|
||
|
}
|
||
|
else {
|
||
|
var contentType = req.getResponseHeader("Content-Type");
|
||
|
if (dwr.engine.isJaxerServer) {
|
||
|
// HACK! Jaxer does something b0rken with Content-Type
|
||
|
contentType = "text/javascript";
|
||
|
}
|
||
|
if (!contentType.match(/^text\/plain/) && !contentType.match(/^text\/javascript/)) {
|
||
|
if (contentType.match(/^text\/html/) && typeof batch.textHtmlHandler == "function") {
|
||
|
batch.textHtmlHandler({ status:status, responseText:reply, contentType:contentType });
|
||
|
}
|
||
|
else {
|
||
|
dwr.engine._handleWarning(batch, { name:"dwr.engine.invalidMimeType", message:"Invalid content type: '" + contentType + "'" });
|
||
|
}
|
||
|
}
|
||
|
else {
|
||
|
// Comet replies might have already partially executed
|
||
|
if (batch.isPoll && batch.map.partialResponse == dwr.engine._partialResponseYes) {
|
||
|
dwr.engine.transport.xhr.processCometResponse(reply, batch);
|
||
|
}
|
||
|
else {
|
||
|
if (reply.search("//#DWR") == -1) {
|
||
|
dwr.engine._handleWarning(batch, { name:"dwr.engine.invalidReply", message:"Invalid reply from server" });
|
||
|
}
|
||
|
else {
|
||
|
toEval = reply;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
catch (ex) {
|
||
|
dwr.engine._handleWarning(batch, ex);
|
||
|
}
|
||
|
|
||
|
dwr.engine._callPostHooks(batch);
|
||
|
|
||
|
// Outside of the try/catch so errors propagate normally:
|
||
|
dwr.engine._receivedBatch = batch;
|
||
|
if (toEval != null) toEval = toEval.replace(dwr.engine._scriptTagProtection, "");
|
||
|
dwr.engine._eval(toEval);
|
||
|
dwr.engine._receivedBatch = null;
|
||
|
dwr.engine.batch.validate(batch);
|
||
|
if (!batch.completed) dwr.engine.batch.remove(batch);
|
||
|
},
|
||
|
|
||
|
/**
|
||
|
* Check for reverse Ajax activity
|
||
|
* @private
|
||
|
*/
|
||
|
checkCometPoll:function() {
|
||
|
if (dwr.engine._pollReq) {
|
||
|
var req = dwr.engine._pollReq;
|
||
|
var text = req.responseText;
|
||
|
if (text != null) {
|
||
|
dwr.engine.transport.xhr.processCometResponse(text, req.batch);
|
||
|
}
|
||
|
setTimeout(dwr.engine.transport.xhr.checkCometPoll, dwr.engine._pollCometInterval);
|
||
|
} else {
|
||
|
// if _pollReq is null, something bad has happened. Calling _poll will initiate the retry policy.
|
||
|
dwr.engine._poll();
|
||
|
}
|
||
|
},
|
||
|
|
||
|
/**
|
||
|
* Some more text might have come in, test and execute the new stuff.
|
||
|
* This method could also be called by the iframe transport
|
||
|
* @private
|
||
|
* @param {Object} response from xhr.responseText
|
||
|
* @param {Object} batch The batch that the XHR object pertains to
|
||
|
*/
|
||
|
processCometResponse:function(response, batch) {
|
||
|
if (batch.charsProcessed == response.length) return;
|
||
|
if (response.length == 0) {
|
||
|
batch.charsProcessed = 0;
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
var firstStartTag = response.indexOf("//#DWR-START#", batch.charsProcessed);
|
||
|
if (firstStartTag == -1) {
|
||
|
// dwr.engine._debug("No start tag (search from " + batch.charsProcessed + "). skipping '" + response.substring(batch.charsProcessed) + "'");
|
||
|
batch.charsProcessed = response.length;
|
||
|
return;
|
||
|
}
|
||
|
// if (firstStartTag > 0) {
|
||
|
// dwr.engine._debug("Start tag not at start (search from " + batch.charsProcessed + "). skipping '" + response.substring(batch.charsProcessed, firstStartTag) + "'");
|
||
|
// }
|
||
|
|
||
|
var lastEndTag = response.lastIndexOf("//#DWR-END#");
|
||
|
if (lastEndTag == -1) {
|
||
|
// dwr.engine._debug("No end tag. unchanged charsProcessed=" + batch.charsProcessed);
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
// Skip the end tag too for next time, remembering CR and LF
|
||
|
if (response.charCodeAt(lastEndTag + 11) == 13 && response.charCodeAt(lastEndTag + 12) == 10) {
|
||
|
batch.charsProcessed = lastEndTag + 13;
|
||
|
}
|
||
|
else {
|
||
|
batch.charsProcessed = lastEndTag + 11;
|
||
|
}
|
||
|
|
||
|
var exec = response.substring(firstStartTag + 13, lastEndTag);
|
||
|
|
||
|
try {
|
||
|
dwr.engine._receivedBatch = batch;
|
||
|
dwr.engine._eval(exec);
|
||
|
dwr.engine._receivedBatch = null;
|
||
|
}
|
||
|
catch (ex) {
|
||
|
// This is one of these annoying points where we might be executing
|
||
|
// while the window is being destroyed. If dwr == null, bail out.
|
||
|
if (dwr != null) {
|
||
|
dwr.engine._handleError(batch, ex);
|
||
|
}
|
||
|
}
|
||
|
},
|
||
|
|
||
|
/**
|
||
|
* Tidy-up when an XHR call is done
|
||
|
* @param {Object} batch
|
||
|
*/
|
||
|
remove:function(batch) {
|
||
|
// XHR tidyup: avoid IE handles increase
|
||
|
if (batch.req) {
|
||
|
// If this is a poll frame then stop comet polling
|
||
|
if (batch.req == dwr.engine._pollReq) dwr.engine._pollReq = null;
|
||
|
delete batch.req;
|
||
|
}
|
||
|
}
|
||
|
},
|
||
|
|
||
|
/**
|
||
|
* Functions for remoting through IFrame
|
||
|
*/
|
||
|
iframe:{
|
||
|
/**
|
||
|
* The default HTTP method to use
|
||
|
*/
|
||
|
httpMethod:"POST",
|
||
|
|
||
|
/**
|
||
|
* Setup a batch for transfer through IFrame
|
||
|
* @param {Object} batch The batch to alter for IFrame transmit
|
||
|
*/
|
||
|
send:function(batch) {
|
||
|
batch.httpMethod = dwr.engine.transport.iframe.httpMethod;
|
||
|
if (batch.fileUpload) {
|
||
|
batch.httpMethod = "POST";
|
||
|
batch.encType = "multipart/form-data";
|
||
|
}
|
||
|
var idname = dwr.engine.transport.iframe.getId(batch);
|
||
|
if (dwr.engine.isIE) {
|
||
|
batch.div = document.createElement("div");
|
||
|
document.body.appendChild(batch.div);
|
||
|
batch.div.innerHTML = "<iframe src='about:blank' frameborder='0' style='width:0px;height:0px;border:0;display:none;' id='" + idname + "' name='" + idname + "'></iframe>";
|
||
|
batch.iframe = batch.div.firstChild;
|
||
|
} else {
|
||
|
batch.iframe = document.createElement("iframe");
|
||
|
batch.iframe.setAttribute("id", idname);
|
||
|
batch.iframe.setAttribute("name", idname);
|
||
|
batch.iframe.setAttribute("frameborder", "0");
|
||
|
batch.iframe.setAttribute("src", "about:blank");
|
||
|
batch.iframe.setAttribute("style", "width:0px;height:0px;border:0;display:none;");
|
||
|
document.body.appendChild(batch.iframe);
|
||
|
}
|
||
|
batch.document = document;
|
||
|
batch.iframe.batch = batch;
|
||
|
dwr.engine.transport.iframe.beginLoader(batch, idname);
|
||
|
},
|
||
|
|
||
|
/**
|
||
|
* Create a unique ID so multiple iframes can fire at the same time
|
||
|
* @param {Object} batch A source of a unique number for the batch
|
||
|
* @return {String} a name prefix for created elements
|
||
|
*/
|
||
|
getId:function(batch) {
|
||
|
return batch.isPoll ? "dwr-if-poll-" + batch.map.batchId : "dwr-if-" + batch.map.batchId;
|
||
|
},
|
||
|
|
||
|
/**
|
||
|
* Setup a form or construct a src attribute to use the iframe.
|
||
|
* This is abstracted from send() because the same logic will do for htmlfile
|
||
|
* @param {Object} batch
|
||
|
*/
|
||
|
beginLoader:function(batch, idname) {
|
||
|
batch.mode = batch.isPoll ? dwr.engine._ModeHtmlPoll : dwr.engine._ModeHtmlCall;
|
||
|
if (batch.isPoll) dwr.engine._outstandingIFrames.push(batch.iframe);
|
||
|
var request = dwr.engine.batch.constructRequest(batch, batch.httpMethod);
|
||
|
if (batch.httpMethod == "GET") {
|
||
|
batch.iframe.setAttribute("src", request.url);
|
||
|
}
|
||
|
else {
|
||
|
// TODO: On firefox we can now get the values of file fields, maybe we should use this
|
||
|
// See http://soakedandsoaped.com/articles/read/firefox-3-native-ajax-file-upload
|
||
|
// setting enctype via the DOM does not work in IE, create the form using innerHTML instead
|
||
|
batch.form = batch.document.createElement("form");
|
||
|
batch.form.setAttribute("id", "dwr-form-" + idname);
|
||
|
batch.form.setAttribute("action", request.url);
|
||
|
batch.form.setAttribute("target", idname);
|
||
|
batch.form.setAttribute("style", "display:none");
|
||
|
batch.form.setAttribute("method", batch.httpMethod);
|
||
|
if (batch.encType) {
|
||
|
batch.form.setAttribute("encType", batch.encType);
|
||
|
batch.form.setAttribute("encoding", batch.encType);
|
||
|
}
|
||
|
for (var prop in batch.map) {
|
||
|
var value = batch.map[prop];
|
||
|
if (typeof value != "function") {
|
||
|
if (value && value.tagName && value.tagName.toLowerCase() == "input" && value.type && value.type.toLowerCase() == "file") {
|
||
|
// Since we can not set the value of a file object, we must post the actual file object
|
||
|
// that the user clicked browse on. We will put a clone in it's place.
|
||
|
var clone = value.cloneNode(true);
|
||
|
value.removeAttribute("id", prop);
|
||
|
value.setAttribute("name", prop);
|
||
|
value.style.display = "none";
|
||
|
value.parentNode.insertBefore(clone, value);
|
||
|
value.parentNode.removeChild(value);
|
||
|
batch.form.appendChild(value);
|
||
|
} else {
|
||
|
var formInput = batch.document.createElement("input");
|
||
|
formInput.setAttribute("type", "hidden");
|
||
|
formInput.setAttribute("name", prop);
|
||
|
formInput.setAttribute("value", value);
|
||
|
batch.form.appendChild(formInput);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
batch.document.body.appendChild(batch.form);
|
||
|
batch.form.submit();
|
||
|
}
|
||
|
},
|
||
|
|
||
|
/**
|
||
|
* Called from iframe onload, check batch using batch-id
|
||
|
* @private
|
||
|
* @param {int} batchId The id of the batch that has loaded
|
||
|
*/
|
||
|
loadingComplete:function(batchId) {
|
||
|
var batch = dwr.engine._batches[batchId];
|
||
|
if (batch) dwr.engine.batch.validate(batch);
|
||
|
},
|
||
|
|
||
|
/**
|
||
|
* Functions designed to be called by the server
|
||
|
*/
|
||
|
remote:{
|
||
|
/**
|
||
|
* Called by the server: An IFrame reply is about to start
|
||
|
* @private
|
||
|
* @param {Object} iframe
|
||
|
* @param {int} batchId
|
||
|
*/
|
||
|
beginIFrameResponse:function(iframe, batchId) {
|
||
|
if (iframe != null) dwr.engine._receivedBatch = iframe.batch;
|
||
|
dwr.engine._callPostHooks(dwr.engine._receivedBatch);
|
||
|
},
|
||
|
|
||
|
/**
|
||
|
* Called by the server: An IFrame reply is just completing
|
||
|
* @private
|
||
|
* @param {int} batchId
|
||
|
*/
|
||
|
endIFrameResponse:function(batchId) {
|
||
|
dwr.engine.transport.iframe.loadingComplete(batchId);
|
||
|
dwr.engine.batch.remove(dwr.engine._receivedBatch);
|
||
|
dwr.engine._receivedBatch = null;
|
||
|
}
|
||
|
},
|
||
|
|
||
|
remove:function(batch) {
|
||
|
if (batch.iframe && batch.iframe.parentNode) {
|
||
|
// Safari 3 and Chrome 1 will show endless loading spinner if removing
|
||
|
// iframe during execution of iframe script, so we delay it a bit
|
||
|
setTimeout(function(){
|
||
|
batch.iframe.parentNode.removeChild(batch.iframe);
|
||
|
batch.iframe = null;
|
||
|
}, 100);
|
||
|
}
|
||
|
if (batch.div && batch.div.parentNode)
|
||
|
batch.div.parentNode.removeChild(batch.div);
|
||
|
if (batch.form && batch.form.parentNode)
|
||
|
batch.form.parentNode.removeChild(batch.form);
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
// If we have an iframe comet solution where we need to read data streamed
|
||
|
// into an iframe then we need code like this to slurp the data out.
|
||
|
// Compare this with xhr.checkCometPoll()
|
||
|
outstandingIFrames:[],
|
||
|
|
||
|
checkCometPoll:function() {
|
||
|
for (var i = 0; i < dwr.engine.transport.iframe.outstandingIFrames.length; i++) {
|
||
|
var text = "";
|
||
|
var iframe = dwr.engine.transport.iframe.outstandingIFrames[i];
|
||
|
try {
|
||
|
text = dwr.engine.transport.iframe.getTextFromCometIFrame(iframe);
|
||
|
}
|
||
|
catch (ex) {
|
||
|
dwr.engine._handleWarning(iframe.batch, ex);
|
||
|
}
|
||
|
if (text != "") dwr.engine.transport.xhr.processCometResponse(text, iframe.batch);
|
||
|
}
|
||
|
|
||
|
if (dwr.engine.transport.iframe.outstandingIFrames.length > 0) {
|
||
|
setTimeout(dwr.engine.transport.iframe.checkCometPoll, dwr.engine._pollCometInterval);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// We probably also need to update dwr.engine.remote.beginIFrameResponse()
|
||
|
// to call checkCometPoll.
|
||
|
|
||
|
// Extract the whole (executed and all) text from the current iframe
|
||
|
getTextFromCometIFrame:function(frameEle) {
|
||
|
var body = frameEle.contentWindow.document.body;
|
||
|
if (body == null) return "";
|
||
|
var text = body.innerHTML;
|
||
|
// We need to prevent IE from stripping line feeds
|
||
|
if (text.indexOf("<PRE>") == 0 || text.indexOf("<pre>") == 0) {
|
||
|
text = text.substring(5, text.length - 7);
|
||
|
}
|
||
|
return text;
|
||
|
};
|
||
|
|
||
|
// And an addition to iframe.remove():
|
||
|
{
|
||
|
if (batch.iframe) {
|
||
|
// If this is a poll frame then stop comet polling
|
||
|
for (var i = 0; i < dwr.engine.transport.iframe.outstandingIFrames.length; i++) {
|
||
|
if (dwr.engine.transport.iframe.outstandingIFrames[i] == batch.iframe) {
|
||
|
dwr.engine.transport.iframe.outstandingIFrames.splice(i, 1);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
*/
|
||
|
},
|
||
|
|
||
|
/**
|
||
|
* Functions for remoting through Script Tags
|
||
|
*/
|
||
|
scriptTag:{
|
||
|
/**
|
||
|
* Setup a batch for transfer through a script tag
|
||
|
* @param {Object} batch The batch to alter for script tag transmit
|
||
|
*/
|
||
|
send:function(batch) {
|
||
|
batch.mode = batch.isPoll ? dwr.engine._ModePlainPoll : dwr.engine._ModePlainCall;
|
||
|
var request = dwr.engine.batch.constructRequest(batch, "GET");
|
||
|
// The best option is DOM manipulation, but this only works after onload
|
||
|
// has completed
|
||
|
if (document.body) {
|
||
|
batch.script = document.createElement("script");
|
||
|
batch.script.id = "dwr-st-" + batch.map.batchId;
|
||
|
batch.script.src = request.url;
|
||
|
batch.script.type = "text/javascript";
|
||
|
document.body.appendChild(batch.script);
|
||
|
}
|
||
|
else {
|
||
|
document.writeln("<scr" + "ipt type='text/javascript' id='dwr-st-" + batch.map["c0-id"] + "' src='" + request.url + "'> </scr" + "ipt>");
|
||
|
}
|
||
|
}
|
||
|
},
|
||
|
|
||
|
/**
|
||
|
* Remoting through IE's htmlfile ActiveX control
|
||
|
*/
|
||
|
htmlfile:{
|
||
|
/**
|
||
|
* Setup a batch for transfer through htmlfile
|
||
|
* @param {Object} batch The batch to alter for htmlfile transmit
|
||
|
*/
|
||
|
send:function(batch) {
|
||
|
var idname = dwr.engine.transport.iframe.getId(batch);
|
||
|
batch.htmlfile = new window.ActiveXObject("htmlfile");
|
||
|
batch.htmlfile.open();
|
||
|
batch.htmlfile.write("<" + "html>");
|
||
|
batch.htmlfile.write("<div><iframe className='wibble' src='javascript:void(0)' id='" + idname + "' name='" + idname + "' onload='dwr.engine.transport.iframe.loadingComplete(" + batch.map.batchId + ");'></iframe></div>");
|
||
|
batch.htmlfile.write("</" + "html>");
|
||
|
batch.htmlfile.close();
|
||
|
batch.htmlfile.parentWindow.dwr = dwr;
|
||
|
batch.document = batch.htmlfile;
|
||
|
|
||
|
dwr.engine.transport.iframe.beginLoader(batch, idname);
|
||
|
}
|
||
|
}
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* Functions to manipulate batches
|
||
|
* @private
|
||
|
*/
|
||
|
dwr.engine.batch = {
|
||
|
/**
|
||
|
* Generate a new standard batch
|
||
|
* @private
|
||
|
*/
|
||
|
create:function() {
|
||
|
var batch = {
|
||
|
async:dwr.engine._async,
|
||
|
charsProcessed:0,
|
||
|
handlers:[],
|
||
|
isPoll:false,
|
||
|
map:{ callCount:0, windowName:window.name },
|
||
|
paramCount:0,
|
||
|
preHooks:[],
|
||
|
postHooks:[],
|
||
|
timeout:dwr.engine._timeout,
|
||
|
errorHandler:dwr.engine._errorHandler,
|
||
|
warningHandler:dwr.engine._warningHandler,
|
||
|
textHtmlHandler:dwr.engine._textHtmlHandler
|
||
|
};
|
||
|
|
||
|
if (dwr.engine._preHook) {
|
||
|
batch.preHooks.push(dwr.engine._preHook);
|
||
|
}
|
||
|
if (dwr.engine._postHook) {
|
||
|
batch.postHooks.push(dwr.engine._postHook);
|
||
|
}
|
||
|
|
||
|
dwr.engine.batch.populateHeadersAndParameters(batch);
|
||
|
return batch;
|
||
|
},
|
||
|
|
||
|
/**
|
||
|
* Generate a new batch for polling
|
||
|
* @private
|
||
|
* @see dwr.engine.batch.create()
|
||
|
*/
|
||
|
createPoll:function() {
|
||
|
var batch = {
|
||
|
async:true,
|
||
|
charsProcessed:0,
|
||
|
handlers:[{
|
||
|
callback:function(pause) {
|
||
|
dwr.engine._retries = 0;
|
||
|
setTimeout(dwr.engine._poll, pause);
|
||
|
}
|
||
|
}],
|
||
|
isPoll:true,
|
||
|
map:{ windowName:window.name },
|
||
|
paramCount:0,
|
||
|
path:dwr.engine._pathToDwrServlet,
|
||
|
preHooks:[],
|
||
|
postHooks:[],
|
||
|
timeout:0,
|
||
|
windowName:window.name,
|
||
|
errorHandler:dwr.engine._retryHandler,
|
||
|
warningHandler:dwr.engine._retryHandler,
|
||
|
textHtmlHandler:dwr.engine._textHtmlHandler
|
||
|
};
|
||
|
|
||
|
dwr.engine.batch.populateHeadersAndParameters(batch);
|
||
|
return batch;
|
||
|
},
|
||
|
|
||
|
/**
|
||
|
* Copy the global headers and parameters into this batch object
|
||
|
* @private
|
||
|
* @param {Object} batch The destination
|
||
|
*/
|
||
|
populateHeadersAndParameters:function(batch) {
|
||
|
var propname, data;
|
||
|
batch.headers = {};
|
||
|
if (dwr.engine._headers) {
|
||
|
for (propname in dwr.engine._headers) {
|
||
|
data = dwr.engine._headers[propname];
|
||
|
if (typeof data != "function") batch.headers[propname] = data;
|
||
|
}
|
||
|
}
|
||
|
batch.attributes = {}; // props to add as request attributes
|
||
|
if (dwr.engine._requestAttributes) {
|
||
|
for (propname in dwr.engine._requestAttributes) {
|
||
|
data = dwr.engine._requestAttributes[propname];
|
||
|
if (typeof data != "function") batch.attributes[propname] = data;
|
||
|
}
|
||
|
}
|
||
|
},
|
||
|
|
||
|
/**
|
||
|
* Augment this batch with a new call
|
||
|
* @private
|
||
|
*/
|
||
|
addCall:function(batch, scriptName, methodName, args) {
|
||
|
// From the other params, work out which is the function (or object with
|
||
|
// call meta-data) and which is the call parameters
|
||
|
var callData, stopAt;
|
||
|
var lastArg = args[args.length - 1];
|
||
|
if (lastArg == null || typeof lastArg == "function") {
|
||
|
callData = { callback:lastArg };
|
||
|
stopAt = args.length - 1;
|
||
|
}
|
||
|
else if (typeof lastArg == "object" && (typeof lastArg.callback == "function"
|
||
|
|| typeof lastArg.exceptionHandler == "function" || typeof lastArg.callbackHandler == "function"
|
||
|
|| typeof lastArg.errorHandler == "function" || typeof lastArg.warningHandler == "function" )) {
|
||
|
callData = lastArg;
|
||
|
stopAt = args.length - 1;
|
||
|
}
|
||
|
else {
|
||
|
callData = {};
|
||
|
stopAt = args.length;
|
||
|
}
|
||
|
|
||
|
// Merge from the callData into the batch
|
||
|
dwr.engine.batch.merge(batch, callData);
|
||
|
batch.handlers[batch.map.callCount] = {
|
||
|
exceptionHandler:callData.exceptionHandler,
|
||
|
exceptionArg:callData.exceptionArg || callData.arg || null,
|
||
|
exceptionScope:callData.exceptionScope || callData.scope || window,
|
||
|
callback:callData.callbackHandler || callData.callback,
|
||
|
callbackArg:callData.callbackArg || callData.arg || null,
|
||
|
callbackScope:callData.callbackScope || callData.scope || window
|
||
|
};
|
||
|
|
||
|
// Copy to the map the things that need serializing
|
||
|
var prefix = "c" + batch.map.callCount + "-";
|
||
|
batch.map[prefix + "scriptName"] = scriptName;
|
||
|
batch.map[prefix + "methodName"] = methodName;
|
||
|
batch.map[prefix + "id"] = batch.map.callCount;
|
||
|
var converted = [];
|
||
|
for (var i = 0; i < stopAt; i++) {
|
||
|
dwr.engine.serialize.convert(batch, converted, args[i], prefix + "param" + i, 0);
|
||
|
}
|
||
|
},
|
||
|
|
||
|
/**
|
||
|
* Take further options and merge them into a batch
|
||
|
* @private
|
||
|
* @param {Object} batch The batch that we are altering
|
||
|
* @param {Object} overrides The object containing properties to copy into batch
|
||
|
*/
|
||
|
merge:function(batch, overrides) {
|
||
|
var propname, data;
|
||
|
for (var i = 0; i < dwr.engine._propnames.length; i++) {
|
||
|
propname = dwr.engine._propnames[i];
|
||
|
if (overrides[propname] != null) batch[propname] = overrides[propname];
|
||
|
}
|
||
|
if (overrides.preHook != null) batch.preHooks.unshift(overrides.preHook);
|
||
|
if (overrides.postHook != null) batch.postHooks.push(overrides.postHook);
|
||
|
if (overrides.headers) {
|
||
|
for (propname in overrides.headers) {
|
||
|
data = overrides.headers[propname];
|
||
|
if (typeof data != "function") batch.headers[propname] = data;
|
||
|
}
|
||
|
}
|
||
|
var reqAttrs = null;
|
||
|
if (overrides.parameters) reqAttrs = overrides.parameters;
|
||
|
if (overrides.requestAttributes) reqAttrs = overrides.requestAttributes;
|
||
|
if (reqAttrs) {
|
||
|
for (propname in reqAttrs) {
|
||
|
data = reqAttrs[propname];
|
||
|
if (typeof data != "function") batch.map["p-" + propname] = "" + data;
|
||
|
}
|
||
|
}
|
||
|
},
|
||
|
|
||
|
/**
|
||
|
* Executed just before a transport sends the batch
|
||
|
* @private
|
||
|
* @param {Object} batch The batch to prepare for sending
|
||
|
*/
|
||
|
prepareToSend:function(batch) {
|
||
|
batch.map.batchId = dwr.engine._nextBatchId;
|
||
|
dwr.engine._nextBatchId++;
|
||
|
dwr.engine._batches[batch.map.batchId] = batch;
|
||
|
dwr.engine._batchesLength++;
|
||
|
batch.completed = false;
|
||
|
|
||
|
// security details are filled in late so previous batches have completed
|
||
|
batch.map.page = encodeURIComponent(window.location.pathname + window.location.search);
|
||
|
batch.map.httpSessionId = dwr.engine._getHttpSessionId();
|
||
|
batch.map.scriptSessionId = dwr.engine._scriptSessionId;
|
||
|
batch.map.windowName = window.name;
|
||
|
|
||
|
for (var i = 0; i < batch.preHooks.length; i++) {
|
||
|
batch.preHooks[i]();
|
||
|
}
|
||
|
batch.preHooks = null;
|
||
|
// Set a timeout
|
||
|
if (batch.timeout && batch.timeout != 0) {
|
||
|
batch.timeoutId = setTimeout(function() { dwr.engine.transport.abort(batch); }, batch.timeout);
|
||
|
}
|
||
|
},
|
||
|
|
||
|
/**
|
||
|
* Work out what the URL should look like
|
||
|
* @private
|
||
|
* @param {Object} batch the data that we are sending
|
||
|
* @param {Object} httpMethod Are we using GET/POST etc?
|
||
|
*/
|
||
|
constructRequest:function(batch, httpMethod) {
|
||
|
// A quick string to help people that use web log analysers
|
||
|
var urlBuffer = [];
|
||
|
urlBuffer.push(batch.path);
|
||
|
urlBuffer.push(batch.mode);
|
||
|
if (batch.isPoll) {
|
||
|
urlBuffer.push("ReverseAjax.dwr");
|
||
|
}
|
||
|
else if (batch.map.callCount == 1) {
|
||
|
urlBuffer.push(batch.map["c0-scriptName"]);
|
||
|
urlBuffer.push(".");
|
||
|
urlBuffer.push(batch.map["c0-methodName"]);
|
||
|
urlBuffer.push(".dwr");
|
||
|
}
|
||
|
else {
|
||
|
urlBuffer.push("Multiple.");
|
||
|
urlBuffer.push(batch.map.callCount);
|
||
|
urlBuffer.push(".dwr");
|
||
|
}
|
||
|
// Play nice with url re-writing
|
||
|
var sessionMatch = location.href.match(/jsessionid=([^?#]+)/);
|
||
|
if (sessionMatch != null) {
|
||
|
urlBuffer.push(";jsessionid=");
|
||
|
urlBuffer.push(sessionMatch[1]);
|
||
|
}
|
||
|
|
||
|
var request = {};
|
||
|
var prop;
|
||
|
if (httpMethod == "GET") {
|
||
|
// Some browsers (Opera/Safari2) seem to fail to convert the callCount value
|
||
|
// to a string in the loop below so we do it manually here.
|
||
|
batch.map.callCount = "" + batch.map.callCount;
|
||
|
urlBuffer.push("?");
|
||
|
for (prop in batch.map) {
|
||
|
if (typeof batch.map[prop] != "function") {
|
||
|
urlBuffer.push(encodeURIComponent(prop));
|
||
|
urlBuffer.push("=");
|
||
|
urlBuffer.push(encodeURIComponent(batch.map[prop]));
|
||
|
urlBuffer.push("&");
|
||
|
}
|
||
|
}
|
||
|
urlBuffer.pop(); // remove the trailing &
|
||
|
request.body = null;
|
||
|
}
|
||
|
else {
|
||
|
// PERFORMANCE: for iframe mode this is thrown away.
|
||
|
if (dwr.engine.isIE <= 7) {
|
||
|
// Use array joining on IE1-7 (fastest)
|
||
|
var buf = [];
|
||
|
for (prop in batch.map) {
|
||
|
if (typeof batch.map[prop] != "function") {
|
||
|
buf.push(prop + "=" + batch.map[prop] + dwr.engine._postSeperator);
|
||
|
}
|
||
|
}
|
||
|
request.body = buf.join("");
|
||
|
}
|
||
|
else {
|
||
|
// Use string concat on other browsers (fastest)
|
||
|
for (prop in batch.map) {
|
||
|
if (typeof batch.map[prop] != "function") {
|
||
|
request.body += prop + "=" + batch.map[prop] + dwr.engine._postSeperator;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
var bodyBuffer = [];
|
||
|
for (prop in batch.map) {
|
||
|
if (typeof batch.map[prop] != "function") {
|
||
|
bodyBuffer.push(prop);
|
||
|
bodyBuffer.push("=");
|
||
|
bodyBuffer.push(batch.map[prop]);
|
||
|
bodyBuffer.push(dwr.engine._postSeperator);
|
||
|
}
|
||
|
}
|
||
|
request.body = dwr.engine._contentRewriteHandler(bodyBuffer.join(""));
|
||
|
}
|
||
|
request.url = dwr.engine._urlRewriteHandler(urlBuffer.join(""));
|
||
|
return request;
|
||
|
},
|
||
|
|
||
|
/**
|
||
|
* @private This function is invoked when a batch reply is received.
|
||
|
* It checks that there is a response for every call in the batch. Otherwise,
|
||
|
* an error will be signaled (a call without a response indicates that the
|
||
|
* server failed to send complete batch response).
|
||
|
*/
|
||
|
validate:function(batch) {
|
||
|
// If some call left unreplied, report an error.
|
||
|
if (!batch.completed) {
|
||
|
for (var i = 0; i < batch.map.callCount; i++) {
|
||
|
if (batch.handlers[i].completed !== true) {
|
||
|
dwr.engine._handleWarning(batch, { name:"dwr.engine.incompleteReply", message:"Incomplete reply from server" });
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
},
|
||
|
|
||
|
/**
|
||
|
* A call has finished by whatever means and we need to shut it all down.
|
||
|
* @private
|
||
|
* @param {Object} batch The batch that we are altering
|
||
|
*/
|
||
|
remove:function(batch) {
|
||
|
if (!batch) {
|
||
|
dwr.engine._debug("Warning: null batch in dwr.engine.batch.remove()", true);
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
if (batch.completed == "true") {
|
||
|
dwr.engine._debug("Warning: Double complete", true);
|
||
|
return;
|
||
|
}
|
||
|
batch.completed = true;
|
||
|
|
||
|
// Transport tidyup
|
||
|
dwr.engine.transport.remove(batch);
|
||
|
|
||
|
// Timeout tidyup
|
||
|
if (batch.timeoutId) {
|
||
|
clearTimeout(batch.timeoutId);
|
||
|
delete batch.timeoutId;
|
||
|
}
|
||
|
|
||
|
// TODO: co-locate all the functions that work on a set of batches
|
||
|
if (batch.map && (batch.map.batchId || batch.map.batchId == 0)) {
|
||
|
delete dwr.engine._batches[batch.map.batchId];
|
||
|
dwr.engine._batchesLength--;
|
||
|
}
|
||
|
|
||
|
// If there is anything on the queue waiting to go out, then send it.
|
||
|
// We don't need to check for ordered mode, here because when ordered mode
|
||
|
// gets turned off, we still process *waiting* batches in an ordered way.
|
||
|
if (dwr.engine._batchQueue.length != 0) {
|
||
|
var sendbatch = dwr.engine._batchQueue.shift();
|
||
|
dwr.engine.transport.send(sendbatch);
|
||
|
}
|
||
|
}
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* Various utility functions
|
||
|
* @private
|
||
|
*/
|
||
|
dwr.engine.util = {
|
||
|
/**
|
||
|
* Create one of a number of ActiveX strings
|
||
|
* @param {Object} axarray An array of strings to attempt to create ActiveX objects from
|
||
|
*/
|
||
|
newActiveXObject:function(axarray) {
|
||
|
var returnValue;
|
||
|
for (var i = 0; i < axarray.length; i++) {
|
||
|
try {
|
||
|
returnValue = new ActiveXObject(axarray[i]);
|
||
|
break;
|
||
|
}
|
||
|
catch (ex) { /* ignore */ }
|
||
|
}
|
||
|
return returnValue;
|
||
|
}
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* Work out what type of browser we are working on
|
||
|
*/
|
||
|
var userAgent = navigator.userAgent;
|
||
|
var versionString = navigator.appVersion;
|
||
|
var version = parseFloat(versionString);
|
||
|
|
||
|
dwr.engine.isOpera = (userAgent.indexOf("Opera") >= 0) ? version : 0;
|
||
|
dwr.engine.isKhtml = (versionString.indexOf("Konqueror") >= 0) || (versionString.indexOf("Safari") >= 0) ? version : 0;
|
||
|
dwr.engine.isSafari = (versionString.indexOf("Safari") >= 0) ? version : 0;
|
||
|
dwr.engine.isJaxerServer = (window.Jaxer && Jaxer.isOnServer);
|
||
|
|
||
|
var geckoPos = userAgent.indexOf("Gecko");
|
||
|
dwr.engine.isMozilla = ((geckoPos >= 0) && (!dwr.engine.isKhtml)) ? version : 0;
|
||
|
|
||
|
dwr.engine.isFF = 0;
|
||
|
dwr.engine.isIE = 0;
|
||
|
try {
|
||
|
if (dwr.engine.isMozilla) {
|
||
|
dwr.engine.isFF = parseFloat(userAgent.split("Firefox/")[1].split(" ")[0]);
|
||
|
}
|
||
|
if ((document.all) && (!dwr.engine.isOpera)) {
|
||
|
dwr.engine.isIE = parseFloat(versionString.split("MSIE ")[1].split(";")[0]);
|
||
|
}
|
||
|
}
|
||
|
catch(ex) { }
|
||
|
|
||
|
// Fetch the scriptSessionId from the server
|
||
|
eval("dwr.engine._execute(dwr.engine._pathToDwrServlet, '__System', 'pageLoaded', [ function() { dwr.engine._ordered = false; }]);");
|
||
|
|
||
|
/**
|
||
|
* Routines for the DWR pubsub hub
|
||
|
*/
|
||
|
dwr.hub = {
|
||
|
/**
|
||
|
* Publish some data to a given topic
|
||
|
* @param {Object} topicName The topic to publish to
|
||
|
* @param {Object} data The data to publish
|
||
|
*/
|
||
|
publish:function(topicName, data) {
|
||
|
dwr.engine._execute(dwr.engine._pathToDwrServlet, '__System', 'publish', topicName, data, {});
|
||
|
},
|
||
|
|
||
|
/**
|
||
|
* Subscribe to get notifications of publish events to a given topic
|
||
|
* @param {String} topicName The topic to subscribe to
|
||
|
* @param {Function} callback The function to call when a publish happens
|
||
|
* @param {Object} scope The 'this' object on which the callback executes (optional)
|
||
|
* @param {Object} subscriberData Data that the subscriber wishes to remember (optional)
|
||
|
* @return An opaque type for use with unsubscribe
|
||
|
*/
|
||
|
subscribe:function(topicName, callback, scope, subscriberData) {
|
||
|
var subscription = "" + dwr.hub._subscriptionId;
|
||
|
dwr.hub._subscriptionId++;
|
||
|
dwr.hub._subscriptions[subscription] = {
|
||
|
callback:callback,
|
||
|
scope:scope,
|
||
|
subscriberData:subscriberData
|
||
|
};
|
||
|
dwr.engine._execute(dwr.engine._pathToDwrServlet, '__System', 'subscribe', topicName, subscription, {});
|
||
|
return subscription;
|
||
|
},
|
||
|
|
||
|
/**
|
||
|
* Called by the server: A publish event has happened that we care about
|
||
|
* @private
|
||
|
* @param {Object} subscriptionId
|
||
|
* @param {Object} publishData
|
||
|
*/
|
||
|
_remotePublish:function(subscriptionId, publishData) {
|
||
|
var subscriptionData = dwr.hub._subscriptions[subscriptionId];
|
||
|
if (!subscriptionData) return;
|
||
|
subscriptionData.callback.call(subscriptionData.scope, publishData, subscriptionData.subscriberData);
|
||
|
},
|
||
|
|
||
|
/**
|
||
|
* Each time we subscribe to something, we use a unique number
|
||
|
*/
|
||
|
_subscriptionId:0,
|
||
|
|
||
|
/**
|
||
|
* We need to remember what we are subscribed to so we can recall the callback
|
||
|
*/
|
||
|
_subscriptions:{}
|
||
|
};
|
||
|
})();
|
||
|
|
||
|
/**
|
||
|
* High level data-sync API for use by Widget libraries like a Dojo-Data-Store.
|
||
|
* For full documentation see org.directwebremoting.export.Data
|
||
|
*/
|
||
|
dwr.data = {
|
||
|
|
||
|
/**
|
||
|
* This is just documentation that defines how the listener parameter must act
|
||
|
* in order to receive asynchronous updates
|
||
|
*/
|
||
|
StoreChangeListener:{
|
||
|
/**
|
||
|
* Something has removed an item from the store
|
||
|
* @param {StoreProvider} source The store from which it was moved
|
||
|
* @param {string} itemId The ID of the item
|
||
|
*/
|
||
|
itemRemoved:function(source, itemId) { },
|
||
|
|
||
|
/**
|
||
|
* Something has added an item to the store
|
||
|
* @param {StoreProvider} source The store from which it was moved
|
||
|
* @param {Item} item The thing that has changed
|
||
|
*/
|
||
|
itemAdded:function(source, item) { },
|
||
|
|
||
|
/**
|
||
|
* Something has updated an item in the store
|
||
|
* @param {StoreProvider} source The store from which it was moved
|
||
|
* @param {Item} item The thing that has changed
|
||
|
* @param {string[]} changedAttributes A list of changed attributes. If null then
|
||
|
* you should assume that everything has changed
|
||
|
*/
|
||
|
itemChanged:function(source, item, changedAttributes) { }
|
||
|
},
|
||
|
|
||
|
/**
|
||
|
* Create a cache object containing the functions to interact with a server
|
||
|
* side StoreProvider
|
||
|
* @param {string} storeId ID of server provided storage
|
||
|
* @param {dwr.data.StoreChangeListener} listener See server documentation
|
||
|
* This is likely to be true if dwr.engine.activeReverseAjax == true
|
||
|
*/
|
||
|
Cache:function(storeId, listener) {
|
||
|
this.storeId = storeId;
|
||
|
this.listener = listener;
|
||
|
}
|
||
|
};
|
||
|
|
||
|
(function() {
|
||
|
/**
|
||
|
* Notes that there is a region of a page that wishes to subscribe to server
|
||
|
* side data and registers a callback function to receive the data.
|
||
|
* @param {Object} region filtering and sorting options. Includes:
|
||
|
* - start: The beginning of the region of specific interest
|
||
|
* - count: The number of items being viewed
|
||
|
* - sort: The sort criteria
|
||
|
* - query: The filter criteria
|
||
|
* @param {function|object} callbackObj A standard DWR callback object
|
||
|
* @return
|
||
|
*/
|
||
|
dwr.data.Cache.prototype.viewRegion = function(region, callbackObj) {
|
||
|
if (!region) region = { };
|
||
|
if (!region.start) region.start = 0;
|
||
|
if (!region.count) region.count = -1;
|
||
|
if (!region.sort) region.sort = [];
|
||
|
else {
|
||
|
for (var index = 0; index < region.sort.length; index++) {
|
||
|
if (typeof region.sort[index].descending == "undefined") {
|
||
|
region.sort[index].descending = false;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
if (!region.query) region.query = {};
|
||
|
|
||
|
return dwr.engine._execute(dwr.engine._pathToDwrServlet, '__Data', 'viewRegion', [ this.storeId, region, this.listener, callbackObj ]);
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* As dwr.data.Cache.viewRegion() except that we only want to see a single item.
|
||
|
* @param {string} itemId ID of object within the given store
|
||
|
* @param {function|object} callbackObj A standard DWR callback object
|
||
|
*/
|
||
|
dwr.data.Cache.prototype.viewItem = function(itemId, callbackObj) {
|
||
|
return dwr.engine._execute(dwr.engine._pathToDwrServlet, '__Data', 'viewItem', [ this.storeId, itemId, this.listener, callbackObj ]);
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* Undo the action of dwr.data.view()
|
||
|
* @param {function|object} callbackObj A standard DWR callback object
|
||
|
*/
|
||
|
dwr.data.Cache.prototype.unsubscribe = function(callbackObj) {
|
||
|
if (this.listener) {
|
||
|
return dwr.engine._execute(dwr.engine._pathToDwrServlet, '__Data', 'unsubscribe', [ this.storeId, this.listener, callbackObj ]);
|
||
|
}
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* Request an update to server side data
|
||
|
* @param {Object} items An array of update descriptions
|
||
|
* @param {function|object} callbackObj A standard DWR callback object
|
||
|
*/
|
||
|
dwr.data.Cache.prototype.update = function(items, callbackObj) {
|
||
|
return dwr.engine._execute(dwr.engine._pathToDwrServlet, '__Data', 'update', [ this.storeId, items, callbackObj ]);
|
||
|
};
|
||
|
|
||
|
})();
|
||
|
|