Record.js
上传用户:dawnssy
上传日期:2022-08-06
资源大小:9345k
文件大小:15k
源码类别:

JavaScript

开发平台:

JavaScript

  1. /*!
  2.  * Ext JS Library 3.1.0
  3.  * Copyright(c) 2006-2009 Ext JS, LLC
  4.  * licensing@extjs.com
  5.  * http://www.extjs.com/license
  6.  */
  7. /**
  8.  * @class Ext.data.Record
  9.  * <p>Instances of this class encapsulate both Record <em>definition</em> information, and Record
  10.  * <em>value</em> information for use in {@link Ext.data.Store} objects, or any code which needs
  11.  * to access Records cached in an {@link Ext.data.Store} object.</p>
  12.  * <p>Constructors for this class are generated by passing an Array of field definition objects to {@link #create}.
  13.  * Instances are usually only created by {@link Ext.data.Reader} implementations when processing unformatted data
  14.  * objects.</p>
  15.  * <p>Note that an instance of a Record class may only belong to one {@link Ext.data.Store Store} at a time.
  16.  * In order to copy data from one Store to another, use the {@link #copy} method to create an exact
  17.  * copy of the Record, and insert the new instance into the other Store.</p>
  18.  * <p>When serializing a Record for submission to the server, be aware that it contains many private
  19.  * properties, and also a reference to its owning Store which in turn holds references to its Records.
  20.  * This means that a whole Record may not be encoded using {@link Ext.util.JSON.encode}. Instead, use the
  21.  * <code>{@link #data}</code> and <code>{@link #id}</code> properties.</p>
  22.  * <p>Record objects generated by this constructor inherit all the methods of Ext.data.Record listed below.</p>
  23.  * @constructor
  24.  * <p>This constructor should not be used to create Record objects. Instead, use {@link #create} to
  25.  * generate a subclass of Ext.data.Record configured with information about its constituent fields.<p>
  26.  * <p><b>The generated constructor has the same signature as this constructor.</b></p>
  27.  * @param {Object} data (Optional) An object, the properties of which provide values for the new Record's
  28.  * fields. If not specified the <code>{@link Ext.data.Field#defaultValue defaultValue}</code>
  29.  * for each field will be assigned.
  30.  * @param {Object} id (Optional) The id of the Record. The id is used by the
  31.  * {@link Ext.data.Store} object which owns the Record to index its collection
  32.  * of Records (therefore this id should be unique within each store). If an
  33.  * <code>id</code> is not specified a <b><code>{@link #phantom}</code></b>
  34.  * Record will be created with an {@link #Record.id automatically generated id}.
  35.  */
  36. Ext.data.Record = function(data, id){
  37.     // if no id, call the auto id method
  38.     this.id = (id || id === 0) ? id : Ext.data.Record.id(this);
  39.     this.data = data || {};
  40. };
  41. /**
  42.  * Generate a constructor for a specific Record layout.
  43.  * @param {Array} o An Array of <b>{@link Ext.data.Field Field}</b> definition objects.
  44.  * The constructor generated by this method may be used to create new Record instances. The data
  45.  * object must contain properties named after the {@link Ext.data.Field field}
  46.  * <b><tt>{@link Ext.data.Field#name}s</tt></b>.  Example usage:<pre><code>
  47. // create a Record constructor from a description of the fields
  48. var TopicRecord = Ext.data.Record.create([ // creates a subclass of Ext.data.Record
  49.     {{@link Ext.data.Field#name name}: 'title', {@link Ext.data.Field#mapping mapping}: 'topic_title'},
  50.     {name: 'author', mapping: 'username', allowBlank: false},
  51.     {name: 'totalPosts', mapping: 'topic_replies', type: 'int'},
  52.     {name: 'lastPost', mapping: 'post_time', type: 'date'},
  53.     {name: 'lastPoster', mapping: 'user2'},
  54.     {name: 'excerpt', mapping: 'post_text', allowBlank: false},
  55.     // In the simplest case, if no properties other than <tt>name</tt> are required,
  56.     // a field definition may consist of just a String for the field name.
  57.     'signature'
  58. ]);
  59. // create Record instance
  60. var myNewRecord = new TopicRecord(
  61.     {
  62.         title: 'Do my job please',
  63.         author: 'noobie',
  64.         totalPosts: 1,
  65.         lastPost: new Date(),
  66.         lastPoster: 'Animal',
  67.         excerpt: 'No way dude!',
  68.         signature: ''
  69.     },
  70.     id // optionally specify the id of the record otherwise {@link #Record.id one is auto-assigned}
  71. );
  72. myStore.{@link Ext.data.Store#add add}(myNewRecord);
  73. </code></pre>
  74.  * @method create
  75.  * @return {function} A constructor which is used to create new Records according
  76.  * to the definition. The constructor has the same signature as {@link #Record}.
  77.  * @static
  78.  */
  79. Ext.data.Record.create = function(o){
  80.     var f = Ext.extend(Ext.data.Record, {});
  81.     var p = f.prototype;
  82.     p.fields = new Ext.util.MixedCollection(false, function(field){
  83.         return field.name;
  84.     });
  85.     for(var i = 0, len = o.length; i < len; i++){
  86.         p.fields.add(new Ext.data.Field(o[i]));
  87.     }
  88.     f.getField = function(name){
  89.         return p.fields.get(name);
  90.     };
  91.     return f;
  92. };
  93. Ext.data.Record.PREFIX = 'ext-record';
  94. Ext.data.Record.AUTO_ID = 1;
  95. Ext.data.Record.EDIT = 'edit';
  96. Ext.data.Record.REJECT = 'reject';
  97. Ext.data.Record.COMMIT = 'commit';
  98. /**
  99.  * Generates a sequential id. This method is typically called when a record is {@link #create}d
  100.  * and {@link #Record no id has been specified}. The returned id takes the form:
  101.  * <tt>&#123;PREFIX}-&#123;AUTO_ID}</tt>.<div class="mdetail-params"><ul>
  102.  * <li><b><tt>PREFIX</tt></b> : String<p class="sub-desc"><tt>Ext.data.Record.PREFIX</tt>
  103.  * (defaults to <tt>'ext-record'</tt>)</p></li>
  104.  * <li><b><tt>AUTO_ID</tt></b> : String<p class="sub-desc"><tt>Ext.data.Record.AUTO_ID</tt>
  105.  * (defaults to <tt>1</tt> initially)</p></li>
  106.  * </ul></div>
  107.  * @param {Record} rec The record being created.  The record does not exist, it's a {@link #phantom}.
  108.  * @return {String} auto-generated string id, <tt>"ext-record-i++'</tt>;
  109.  */
  110. Ext.data.Record.id = function(rec) {
  111.     rec.phantom = true;
  112.     return [Ext.data.Record.PREFIX, '-', Ext.data.Record.AUTO_ID++].join('');
  113. };
  114. Ext.data.Record.prototype = {
  115.     /**
  116.      * <p><b>This property is stored in the Record definition's <u>prototype</u></b></p>
  117.      * A MixedCollection containing the defined {@link Ext.data.Field Field}s for this Record.  Read-only.
  118.      * @property fields
  119.      * @type Ext.util.MixedCollection
  120.      */
  121.     /**
  122.      * An object hash representing the data for this Record. Every field name in the Record definition
  123.      * is represented by a property of that name in this object. Note that unless you specified a field
  124.      * with {@link Ext.data.Field#name name} "id" in the Record definition, this will <b>not</b> contain
  125.      * an <tt>id</tt> property.
  126.      * @property data
  127.      * @type {Object}
  128.      */
  129.     /**
  130.      * The unique ID of the Record {@link #Record as specified at construction time}.
  131.      * @property id
  132.      * @type {Object}
  133.      */
  134.     /**
  135.      * <p><b>Only present if this Record was created by an {@link Ext.data.XmlReader XmlReader}</b>.</p>
  136.      * <p>The XML element which was the source of the data for this Record.</p>
  137.      * @property node
  138.      * @type {XMLElement}
  139.      */
  140.     /**
  141.      * <p><b>Only present if this Record was created by an {@link Ext.data.ArrayReader ArrayReader} or a {@link Ext.data.JsonReader JsonReader}</b>.</p>
  142.      * <p>The Array or object which was the source of the data for this Record.</p>
  143.      * @property json
  144.      * @type {Array|Object}
  145.      */
  146.     /**
  147.      * Readonly flag - true if this Record has been modified.
  148.      * @type Boolean
  149.      */
  150.     dirty : false,
  151.     editing : false,
  152.     error : null,
  153.     /**
  154.      * This object contains a key and value storing the original values of all modified
  155.      * fields or is null if no fields have been modified.
  156.      * @property modified
  157.      * @type {Object}
  158.      */
  159.     modified : null,
  160.     /**
  161.      * <tt>true</tt> when the record does not yet exist in a server-side database (see
  162.      * {@link #markDirty}).  Any record which has a real database pk set as its id property
  163.      * is NOT a phantom -- it's real.
  164.      * @property phantom
  165.      * @type {Boolean}
  166.      */
  167.     phantom : false,
  168.     // private
  169.     join : function(store){
  170.         /**
  171.          * The {@link Ext.data.Store} to which this Record belongs.
  172.          * @property store
  173.          * @type {Ext.data.Store}
  174.          */
  175.         this.store = store;
  176.     },
  177.     /**
  178.      * Set the {@link Ext.data.Field#name named field} to the specified value.  For example:
  179.      * <pre><code>
  180. // record has a field named 'firstname'
  181. var Employee = Ext.data.Record.{@link #create}([
  182.     {name: 'firstname'},
  183.     ...
  184. ]);
  185. // update the 2nd record in the store:
  186. var rec = myStore.{@link Ext.data.Store#getAt getAt}(1);
  187. // set the value (shows dirty flag):
  188. rec.set('firstname', 'Betty');
  189. // commit the change (removes dirty flag):
  190. rec.{@link #commit}();
  191. // update the record in the store, bypass setting dirty flag,
  192. // and do not store the change in the {@link Ext.data.Store#getModifiedRecords modified records}
  193. rec.{@link #data}['firstname'] = 'Wilma'; // updates record, but not the view
  194. rec.{@link #commit}(); // updates the view
  195.      * </code></pre>
  196.      * <b>Notes</b>:<div class="mdetail-params"><ul>
  197.      * <li>If the store has a writer and <code>autoSave=true</code>, each set()
  198.      * will execute an XHR to the server.</li>
  199.      * <li>Use <code>{@link #beginEdit}</code> to prevent the store's <code>update</code>
  200.      * event firing while using set().</li>
  201.      * <li>Use <code>{@link #endEdit}</code> to have the store's <code>update</code>
  202.      * event fire.</li>
  203.      * </ul></div>
  204.      * @param {String} name The {@link Ext.data.Field#name name of the field} to set.
  205.      * @param {String/Object/Array} value The value to set the field to.
  206.      */
  207.     set : function(name, value){
  208.         var encode = Ext.isPrimitive(value) ? String : Ext.encode;
  209.         if(encode(this.data[name]) == encode(value)) {
  210.             return;
  211.         }        
  212.         this.dirty = true;
  213.         if(!this.modified){
  214.             this.modified = {};
  215.         }
  216.         if(this.modified[name] === undefined){
  217.             this.modified[name] = this.data[name];
  218.         }
  219.         this.data[name] = value;
  220.         if(!this.editing){
  221.             this.afterEdit();
  222.         }
  223.     },
  224.     // private
  225.     afterEdit : function(){
  226.         if(this.store){
  227.             this.store.afterEdit(this);
  228.         }
  229.     },
  230.     // private
  231.     afterReject : function(){
  232.         if(this.store){
  233.             this.store.afterReject(this);
  234.         }
  235.     },
  236.     // private
  237.     afterCommit : function(){
  238.         if(this.store){
  239.             this.store.afterCommit(this);
  240.         }
  241.     },
  242.     /**
  243.      * Get the value of the {@link Ext.data.Field#name named field}.
  244.      * @param {String} name The {@link Ext.data.Field#name name of the field} to get the value of.
  245.      * @return {Object} The value of the field.
  246.      */
  247.     get : function(name){
  248.         return this.data[name];
  249.     },
  250.     /**
  251.      * Begin an edit. While in edit mode, no events (e.g.. the <code>update</code> event)
  252.      * are relayed to the containing store.
  253.      * See also: <code>{@link #endEdit}</code> and <code>{@link #cancelEdit}</code>.
  254.      */
  255.     beginEdit : function(){
  256.         this.editing = true;
  257.         this.modified = this.modified || {};
  258.     },
  259.     /**
  260.      * Cancels all changes made in the current edit operation.
  261.      */
  262.     cancelEdit : function(){
  263.         this.editing = false;
  264.         delete this.modified;
  265.     },
  266.     /**
  267.      * End an edit. If any data was modified, the containing store is notified
  268.      * (ie, the store's <code>update</code> event will fire).
  269.      */
  270.     endEdit : function(){
  271.         this.editing = false;
  272.         if(this.dirty){
  273.             this.afterEdit();
  274.         }
  275.     },
  276.     /**
  277.      * Usually called by the {@link Ext.data.Store} which owns the Record.
  278.      * Rejects all changes made to the Record since either creation, or the last commit operation.
  279.      * Modified fields are reverted to their original values.
  280.      * <p>Developers should subscribe to the {@link Ext.data.Store#update} event
  281.      * to have their code notified of reject operations.</p>
  282.      * @param {Boolean} silent (optional) True to skip notification of the owning
  283.      * store of the change (defaults to false)
  284.      */
  285.     reject : function(silent){
  286.         var m = this.modified;
  287.         for(var n in m){
  288.             if(typeof m[n] != "function"){
  289.                 this.data[n] = m[n];
  290.             }
  291.         }
  292.         this.dirty = false;
  293.         delete this.modified;
  294.         this.editing = false;
  295.         if(silent !== true){
  296.             this.afterReject();
  297.         }
  298.     },
  299.     /**
  300.      * Usually called by the {@link Ext.data.Store} which owns the Record.
  301.      * Commits all changes made to the Record since either creation, or the last commit operation.
  302.      * <p>Developers should subscribe to the {@link Ext.data.Store#update} event
  303.      * to have their code notified of commit operations.</p>
  304.      * @param {Boolean} silent (optional) True to skip notification of the owning
  305.      * store of the change (defaults to false)
  306.      */
  307.     commit : function(silent){
  308.         this.dirty = false;
  309.         delete this.modified;
  310.         this.editing = false;
  311.         if(silent !== true){
  312.             this.afterCommit();
  313.         }
  314.     },
  315.     /**
  316.      * Gets a hash of only the fields that have been modified since this Record was created or commited.
  317.      * @return Object
  318.      */
  319.     getChanges : function(){
  320.         var m = this.modified, cs = {};
  321.         for(var n in m){
  322.             if(m.hasOwnProperty(n)){
  323.                 cs[n] = this.data[n];
  324.             }
  325.         }
  326.         return cs;
  327.     },
  328.     // private
  329.     hasError : function(){
  330.         return this.error !== null;
  331.     },
  332.     // private
  333.     clearError : function(){
  334.         this.error = null;
  335.     },
  336.     /**
  337.      * Creates a copy (clone) of this Record.
  338.      * @param {String} id (optional) A new Record id, defaults to the id
  339.      * of the record being copied. See <code>{@link #id}</code>. 
  340.      * To generate a phantom record with a new id use:<pre><code>
  341. var rec = record.copy(); // clone the record
  342. Ext.data.Record.id(rec); // automatically generate a unique sequential id
  343.      * </code></pre>
  344.      * @return {Record}
  345.      */
  346.     copy : function(newId) {
  347.         return new this.constructor(Ext.apply({}, this.data), newId || this.id);
  348.     },
  349.     /**
  350.      * Returns <tt>true</tt> if the passed field name has been <code>{@link #modified}</code>
  351.      * since the load or last commit.
  352.      * @param {String} fieldName {@link Ext.data.Field.{@link Ext.data.Field#name}
  353.      * @return {Boolean}
  354.      */
  355.     isModified : function(fieldName){
  356.         return !!(this.modified && this.modified.hasOwnProperty(fieldName));
  357.     },
  358.     /**
  359.      * By default returns <tt>false</tt> if any {@link Ext.data.Field field} within the
  360.      * record configured with <tt>{@link Ext.data.Field#allowBlank} = false</tt> returns
  361.      * <tt>true</tt> from an {@link Ext}.{@link Ext#isEmpty isempty} test.
  362.      * @return {Boolean}
  363.      */
  364.     isValid : function() {
  365.         return this.fields.find(function(f) {
  366.             return (f.allowBlank === false && Ext.isEmpty(this.data[f.name])) ? true : false;
  367.         },this) ? false : true;
  368.     },
  369.     /**
  370.      * <p>Marks this <b>Record</b> as <code>{@link #dirty}</code>.  This method
  371.      * is used interally when adding <code>{@link #phantom}</code> records to a
  372.      * {@link Ext.data.Store#writer writer enabled store}.</p>
  373.      * <br><p>Marking a record <code>{@link #dirty}</code> causes the phantom to
  374.      * be returned by {@link Ext.data.Store#getModifiedRecords} where it will
  375.      * have a create action composed for it during {@link Ext.data.Store#save store save}
  376.      * operations.</p>
  377.      */
  378.     markDirty : function(){
  379.         this.dirty = true;
  380.         if(!this.modified){
  381.             this.modified = {};
  382.         }
  383.         this.fields.each(function(f) {
  384.             this.modified[f.name] = this.data[f.name];
  385.         },this);
  386.     }
  387. };