/**
 * The JsonP proxy is useful when you need to load data from a domain other than the one your
 * application is running on. If your application is running on http://domainA.com it cannot use
 * {@link Ext.data.proxy.Ajax Ajax} to load its data from http://domainB.com because cross-domain
 * Ajax requests are prohibited by the browser.
 *
 * We can get around this using a JsonP proxy. JsonP proxy injects a `<script>` tag into the DOM
 * whenever an AJAX request would usually be made. Let's say we want to load data from
 * http://domainB.com/users - the script tag that would be injected might look like this:
 *
 *     <script src="http://domainB.com/users?callback=someCallback"></script>
 *
 * When we inject the tag above, the browser makes a request to that url and includes the response
 * as if it was any other type of JavaScript include. By passing a callback in the url above,
 * we're telling domainB's server that we want to be notified when the result comes in and that it
 * should call our callback function with the data it sends back. So long as the server formats
 * the response to look like this, everything will work:
 *
 *     someCallback({
 *         users: [
 *             {
 *                 id: 1,
 *                 name: "Ed Spencer",
 *                 email: "ed@sencha.com"
 *             }
 *         ]
 *     });
 *
 * As soon as the script finishes loading, the 'someCallback' function that we passed in the url
 * is called with the JSON object that the server returned.
 *
 * JsonP proxy takes care of all of this automatically. It formats the url you pass, adding
 * the callback parameter automatically. It even creates a temporary callback function, waits for it
 * to be called and then puts the data into the Proxy making it look just like you loaded it
 * through a normal {@link Ext.data.proxy.Ajax AjaxProxy}. Here's how we might set that up:
 *
 *     Ext.define('User', {
 *         extend: 'Ext.data.Model',
 *         fields: ['id', 'name', 'email']
 *     });
 *
 *     var store = Ext.create('Ext.data.Store', {
 *         model: 'User',
 *         proxy: {
 *             type: 'jsonp',
 *             url: 'http://domainB.com/users'
 *         }
 *     });
 *
 *     store.load();
 *
 * That's all we need to do - JsonP proxy takes care of the rest. In this case the Proxy will have
 * injected a script tag like this:
 *
 *     <script src="http://domainB.com/users?callback=callback1"></script>
 *
 * # Customization
 *
 * This script tag can be customized using the {@link #callbackKey} configuration. For example:
 *
 *     var store = Ext.create('Ext.data.Store', {
 *         model: 'User',
 *         proxy: {
 *             type: 'jsonp',
 *             url : 'http://domainB.com/users',
 *             callbackKey: 'theCallbackFunction'
 *         }
 *     });
 *
 *     store.load();
 *
 * Would inject a script tag like this:
 *
 *     <script src="http://domainB.com/users?theCallbackFunction=callback1"></script>
 *
 * # Implementing on the server side
 *
 * The remote server side needs to be configured to return data in this format. Here are suggestions
 * for how you might achieve this using Java, PHP and ASP.net:
 *
 * Java:
 *
 *     boolean jsonP = false;
 *     String cb = request.getParameter("callback");
 *     if (cb != null) {
 *         jsonP = true;
 *         response.setContentType("text/javascript");
 *     } else {
 *         response.setContentType("application/x-json");
 *     }
 *     Writer out = response.getWriter();
 *     if (jsonP) {
 *         out.write(cb + "(");
 *     }
 *     out.print(dataBlock.toJsonString());
 *     if (jsonP) {
 *         out.write(");");
 *     }
 *
 * PHP:
 *
 *     $callback = $_REQUEST['callback'];
 *
 *     // Create the output object.
 *     $output = array('a' => 'Apple', 'b' => 'Banana');
 *
 *     //start output
 *     if ($callback) {
 *         header('Content-Type: text/javascript');
 *         echo $callback . '(' . json_encode($output) . ');';
 *     } else {
 *         header('Content-Type: application/x-json');
 *         echo json_encode($output);
 *     }
 *
 * ASP.net:
 *
 *     String jsonString = "{"success": true}";
 *     String cb = Request.Params.Get("callback");
 *     String responseString = "";
 *     if (!String.IsNullOrEmpty(cb)) {
 *         responseString = cb + "(" + jsonString + ")";
 *     } else {
 *         responseString = jsonString;
 *     }
 *     Response.Write(responseString);
 */
Ext.define('Ext.data.proxy.JsonP', {
    extend: 'Ext.data.proxy.Server',
    alternateClassName: 'Ext.data.ScriptTagProxy',
    alias: ['proxy.jsonp', 'proxy.scripttag'],
    requires: ['Ext.data.JsonP'],
 
    config: {
        /**
         * @cfg {String} callbackKey
         * See {@link Ext.data.JsonP#callbackKey}.
         */
        callbackKey: 'callback',
 
        /**
        * @cfg {String} [recordParam]
        * The HTTP parameter name to use when passing records to the server and the
        * {@link #writer Json writer} is not configured to
        * {@link Ext.data.writer.Json#encode encode} records into a parameter.
        * 
        * The {@link #encodeRecords} method is used to encode the records to create this parameter's
        * value.
        */
        recordParam: 'records',
 
        /**
        * @cfg {Boolean} autoAppendParams
        * True to automatically append the request's params to the generated url. Defaults to true
        */
        autoAppendParams: true,
 
        /**
         * @cfg {"name"/"array"/"indexed"} arrayUrlEncodingFormat
         * Format types of {@link #recordParam}.  Valid values are:
         *
         * - **"name"** - Pass records as multiple query {@link #recordParam} params.
         *
         *     ?records=1&records=2&records=3
         *
         * - **"array"** - Pass records as an array using {@link #recordParam} param.
         *
         *     ?records=[1,2,3]
         *
         * - **"indexed"** - Pass records as multiple indexed query {@link recordParam} params.
         *
         *     ?records[0]=1&record[1]=2&records[3]=3
         *
         */
        arrayUrlEncodingFormat: 'name'
    },
 
    /**
     * @private
     * Performs the read request to the remote domain. JsonP proxy does not actually create an Ajax
     * request, instead we write out a `<script>` tag based on the configuration of the internal
     * Ext.data.Request object
     * @param {Ext.data.operation.Operation} operation The
     * {@link Ext.data.operation.Operation Operation} object to execute
     * @param {Function} operation.callback A callback function to execute when the Operation has
     * been completed
     * @param {Object} operation.scope The scope to execute the callback in
     */
    doRequest: function(operation) {
        // generate the unique IDs for this request
        var me = this,
            request = me.buildRequest(operation),
            params = request.getParams();
 
        // apply JsonP proxy-specific attributes to the Request
        request.setConfig({
            callbackKey: me.callbackKey,
            timeout: me.timeout,
            scope: me,
            disableCaching: false, // handled by the proxy
            callback: me.createRequestCallback(request, operation)
        });
 
        // If we are responsible for appending the params to the URL, clear them now so that
        // The Ext.data.JsonP singleton does not append them.
        if (me.getAutoAppendParams()) {
            request.setParams({});
        }
 
        request.setRawRequest(Ext.data.JsonP.request(request.getCurrentConfig()));
 
        // Set the params back once we have made the request though
        request.setParams(params);
        me.lastRequest = request;
 
        return request;
    },
 
    /**
     * @private
     * Creates and returns the function that is called when the request has completed. The returned
     * function should accept a Response object, which contains the response to be read by the
     * configured Reader. The third argument is the callback that should be called after the
     * request has been completed and the Reader has decoded the response. This callback will
     * typically be the callback passed by a store, e.g. in
     * proxy.read(operation, theCallback, scope) theCallback refers to the callback argument
     * received by this function.
     * See {@link #doRequest} for details.
     * @param {Ext.data.Request} request The Request object
     * @param {Ext.data.operation.Operation} operation The Operation being executed
     * @param {Function} operation.callback The callback function to be called when the request
     * completes. This is usually the callback passed to doRequest
     * @param {Object} operation.scope The scope in which to execute the callback function
     * @return {Function} The callback function
     */
    createRequestCallback: function(request, operation) {
        var me = this;
 
        return function(success, response, errorType) {
            if (request === me.lastRequest) {
                me.lastRequest = null;
            }
 
            me.processResponse(success, operation, request, response);
        };
    },
 
    setException: function(operation, response) {
        operation.setException(operation.getRequest().getRawRequest().errorType);
    },
 
    /**
     * Generates a url based on a given Ext.data.Request object. Adds the params and callback
     * function name to the url
     * @param {Ext.data.Request} request The request object
     * @return {String} The url
     */
    buildUrl: function(request) {
        var me = this,
            url = me.callParent(arguments),
            records = request.getRecords(),
            writer = me.getWriter(),
            params,
            filters,
            filter, i, v;
 
        // In the JsonP proxy, params may only go into the URL.
        // So params created by the Writer get applied to the request's params here
        if (writer && request.getOperation().allowWrite()) {
            request = writer.write(request);
        }
 
        // Encode filters into the URL via params
        params = request.getParams();
        filters = params.filters;
        delete params.filters;
 
        if (filters && filters.length) {
            for (= 0; i < filters.length; i++) {
                filter = filters[i];
 
                v = filter.getValue();
 
                if (v) {
                    params[filter.getProperty()] = v;
                }
            }
        }
 
        // If there's no writer, or the writer is not configured to encode the records into
        // a parameter, then we have to do it here.
        if (Ext.isArray(records) && records.length > 0 && (!writer || !writer.getEncode())) {
            params[me.getRecordParam()] = me.encodeRecords(records);
        }
 
        // If we are responsible for appending the params to the URL, do it now.
        // The params are cleared in doRequest so that the Ext.data.JsonP singleton does not
        // add them.
        if (me.getAutoAppendParams()) {
            url = Ext.urlAppend(url,
                                Ext.Object.toQueryString(params,
                                                         me.arrayUrlEncodingFormat === "indexed"));
        }
 
        return url;
    },
 
    /**
     * Aborts a server request. If no request is passed, the most recent request
     * will be aborted.
     * @param {Ext.data.Request} [request] The request to abort.
     */
    abort: function(request) {
        request = request || this.lastRequest;
 
        if (request) {
            Ext.data.JsonP.abort(request.getRawRequest());
        }
    },
 
    /**
     * Encodes an array of records into a value suitable to be added to the request `params`
     * as the {@link #recordParam} parameter. This is broken out into its own function so that
     * it can be easily overridden.
     * 
     * The default implementation 
     * @param {Ext.data.Model[]} records The records array
     * @return {Array} An array of record data objects
     */
    encodeRecords: function(records) {
        var recs = [],
            i = 0,
            len = records.length,
            encodeArray = this.arrayUrlEncodingFormat === "array",
            data;
 
        for (; i < len; i++) {
            data = records[i].getData();
            recs.push(encodeArray ? data : Ext.encode(data));
        }
 
        return encodeArray ? Ext.encode(recs) : recs;
    }
});