/*!
 * Ext JS Library 3.0.3
 * Copyright(c) 2006-2009 Ext JS, LLC
 * licensing@extjs.com
 * http://www.extjs.com/license
 */
/** * @class Ext.data.DataReader * Abstract base class for reading structured data from a data source and converting * it into an object containing {@link Ext.data.Record} objects and metadata for use * by an {@link Ext.data.Store}. This class is intended to be extended and should not * be created directly. For existing implementations, see {@link Ext.data.ArrayReader}, * {@link Ext.data.JsonReader} and {@link Ext.data.XmlReader}. * @constructor Create a new DataReader * @param {Object} meta Metadata configuration options (implementation-specific). * @param {Array/Object} recordType *

Either an Array of {@link Ext.data.Field Field} definition objects (which * will be passed to {@link Ext.data.Record#create}, or a {@link Ext.data.Record Record} * constructor created using {@link Ext.data.Record#create}.

*/ Ext.data.DataReader = function(meta, recordType){
/** * This DataReader's configured metadata as passed to the constructor. * @type Mixed * @property meta */ this.meta = meta;
/** * @cfg {Array/Object} fields *

Either an Array of {@link Ext.data.Field Field} definition objects (which * will be passed to {@link Ext.data.Record#create}, or a {@link Ext.data.Record Record} * constructor created from {@link Ext.data.Record#create}.

*/ this.recordType = Ext.isArray(recordType) ? Ext.data.Record.create(recordType) : recordType; // if recordType defined make sure extraction functions are defined if (this.recordType){ this.buildExtractors(); } }; Ext.data.DataReader.prototype = {
/** * @cfg {String} messageProperty [undefined] Optional name of a property within a server-response that represents a user-feedback message. */
/** * Abstract method created in extension's buildExtractors impl. */ getTotal: Ext.emptyFn,
/** * Abstract method created in extension's buildExtractors impl. */ getRoot: Ext.emptyFn,
/** * Abstract method created in extension's buildExtractors impl. */ getMessage: Ext.emptyFn,
/** * Abstract method created in extension's buildExtractors impl. */ getSuccess: Ext.emptyFn,
/** * Abstract method created in extension's buildExtractors impl. */ getId: Ext.emptyFn,
/** * Abstract method, overridden in DataReader extensions such as {@link Ext.data.JsonReader} and {@link Ext.data.XmlReader} */ buildExtractors : Ext.emptyFn,
/** * Abstract method overridden in DataReader extensions such as {@link Ext.data.JsonReader} and {@link Ext.data.XmlReader} */ extractData : Ext.emptyFn,
/** * Abstract method overridden in DataReader extensions such as {@link Ext.data.JsonReader} and {@link Ext.data.XmlReader} */ extractValues : Ext.emptyFn,
/** * Used for un-phantoming a record after a successful database insert. Sets the records pk along with new data from server. * You must return at least the database pk using the idProperty defined in your DataReader configuration. The incoming * data from server will be merged with the data in the local record. * In addition, you must return record-data from the server in the same order received. * Will perform a commit as well, un-marking dirty-fields. Store's "update" event will be suppressed. * @param {Record/Record[]} record The phantom record to be realized. * @param {Object/Object[]} data The new record data to apply. Must include the primary-key from database defined in idProperty field. */ realize: function(rs, data){ if (Ext.isArray(rs)) { for (var i = rs.length - 1; i >= 0; i--) { // recurse if (Ext.isArray(data)) { this.realize(rs.splice(i,1).shift(), data.splice(i,1).shift()); } else { // weird...rs is an array but data isn't?? recurse but just send in the whole invalid data object. // the else clause below will detect !this.isData and throw exception. this.realize(rs.splice(i,1).shift(), data); } } } else { // If rs is NOT an array but data IS, see if data contains just 1 record. If so extract it and carry on. if (Ext.isArray(data) && data.length == 1) { data = data.shift(); } if (!this.isData(data)) { // TODO: Let exception-handler choose to commit or not rather than blindly rs.commit() here. //rs.commit(); throw new Ext.data.DataReader.Error('realize', rs); } rs.phantom = false; // <-- That's what it's all about rs._phid = rs.id; // <-- copy phantom-id -> _phid, so we can remap in Store#onCreateRecords rs.id = this.getId(data); rs.data = data; rs.commit(); } },
/** * Used for updating a non-phantom or "real" record's data with fresh data from server after remote-save. * If returning data from multiple-records after a batch-update, you must return record-data from the server in * the same order received. Will perform a commit as well, un-marking dirty-fields. Store's "update" event will be * suppressed as the record receives fresh new data-hash * @param {Record/Record[]} rs * @param {Object/Object[]} data */ update : function(rs, data) { if (Ext.isArray(rs)) { for (var i=rs.length-1; i >= 0; i--) { if (Ext.isArray(data)) { this.update(rs.splice(i,1).shift(), data.splice(i,1).shift()); } else { // weird...rs is an array but data isn't?? recurse but just send in the whole data object. // the else clause below will detect !this.isData and throw exception. this.update(rs.splice(i,1).shift(), data); } } } else { // If rs is NOT an array but data IS, see if data contains just 1 record. If so extract it and carry on. if (Ext.isArray(data) && data.length == 1) { data = data.shift(); } if (this.isData(data)) { rs.data = Ext.apply(rs.data, data); } rs.commit(); } },
/** * Returns true if the supplied data-hash looks and quacks like data. Checks to see if it has a key * corresponding to idProperty defined in your DataReader config containing non-empty pk. * @param {Object} data * @return {Boolean} */ isData : function(data) { return (data && Ext.isObject(data) && !Ext.isEmpty(this.getId(data))) ? true : false; }, // private function a store will createSequence upon onMetaChange : function(meta){ delete this.ef; this.meta = meta; this.recordType = Ext.data.Record.create(meta.fields); this.buildExtractors(); } };
/** * @class Ext.data.DataReader.Error * @extends Ext.Error * General error class for Ext.data.DataReader */ Ext.data.DataReader.Error = Ext.extend(Ext.Error, { constructor : function(message, arg) { this.arg = arg; Ext.Error.call(this, message); }, name: 'Ext.data.DataReader' }); Ext.apply(Ext.data.DataReader.Error.prototype, { lang : { 'update': "#update received invalid data from server. Please see docs for DataReader#update and review your DataReader configuration.", 'realize': "#realize was called with invalid remote-data. Please see the docs for DataReader#realize and review your DataReader configuration.", 'invalid-response': "#readResponse received an invalid response from the server." } });
/** * Ext.data.Response * A generic response class to normalize response-handling internally to the framework. * TODO move to own file, add to jsb. */ Ext.data.Response = function(params) { Ext.apply(this, params); }; Ext.data.Response.prototype = {
/** * @property {String} action {@link Ext.data.Api#actions} */ action: undefined,
/** * @property {Boolean} success */ success : undefined,
/** * @property {String} message */ message : undefined,
/** * @property {Array/Object} data */ data: undefined,
/** * @property {Object} raw The raw response returned from server-code */ raw: undefined,
/** * @property {Ext.data.Record/Ext.data.Record[]} record(s) related to the Request action */ records: undefined }