/**
 * Base class for data object.
 */
Ext.define('Ext.exporter.data.Base', {
    requires: [
        'Ext.util.Collection'
    ],
 
    config: {
        /**
         * @cfg {String} idPrefix 
         *
         * Prefix to use when generating the id.
         *
         * @private
         */
        idPrefix: 'id',
        /**
         * @cfg {String} id 
         *
         * Unique id for this object. Auto generated when missing.
         */
        id: null,
        /**
         * @cfg {Boolean} autoGenerateId 
         *
         * Set to `true` to auto generate an id if none is defined.
         */
        autoGenerateId: true
    },
 
    // keep references to internal collections to easily destroy them 
    internalCols: null,
    clearPropertiesOnDestroy: false,
 
    constructor: function(config){
        var me = this;
 
        me.internalCols = [];
        me.initConfig(config);
 
        if(!me._id){
            me.setId(null);
        }
        return me.callParent([config]);
    },
 
    destroy: function(){
        this.destroyCollections();
        this.callParent();
        this.internalCols = null;
    },
 
    destroyCollections: function(){
        var cols = this.internalCols,
            len = cols.length,
            i, j, length, col;
 
        for(= 0; i < len; i++){
            col = cols[i];
            length = col.length;
            for(= 0; j < length; j++) {
                col.items[j].destroy();
            }
            col.destroy();
        }
        cols.length = 0;
    },
 
    clearCollections: function(cols){
        var i, len, col;
 
        cols = cols ? Ext.Array.from(cols) : this.internalCols;
        len = cols.length;
 
        for(= len - 1; i >= 0; i--){
            col = cols[i];
            if(col) {
                col.destroy();
            }
            Ext.Array.remove(this.internalCols, col);
        }
    },
 
    applyId: function(data){
        var id;
 
        if(!data && this._autoGenerateId){
            id = this._idPrefix + (++Ext.idSeed);
        }else{
            id = data;
        }
 
        return id;
    },
 
    /**
     * This method could be used in config appliers that need to initialize a
     * Collection that has items of type className.
     *
     * @param data
     * @param dataCollection
     * @param className
     * @return {Ext.util.Collection}
     */
    checkCollection: function(data, dataCollection, className){
        var col;
 
        if(data){
            col = this.constructCollection(className);
            col.add(data);
        }
 
        if(dataCollection){
            Ext.Array.remove(this.internalCols, dataCollection);
            Ext.destroy(dataCollection.items, dataCollection);
        }
 
        return col;
    },
 
    /**
     * Create a new Collection with a decoder for the specified className.
     *
     * @param className
     * @returns {Ext.util.Collection}
     *
     * @private
     */
    constructCollection: function(className){
        var cls = Ext.ClassManager.get(className),
            cfg = {
                decoder: this.getCollectionDecoder(cls)
            },
            col;
 
        if(typeof cls.getKey === 'function'){
            cfg.keyFn = this.getCollectionItemKey;
        }
 
        col = new Ext.util.Collection(cfg);
 
        this.internalCols.push(col);
        return col;
    },
 
    /**
     * Builds a Collection decoder for the specified className.
     *
     * @param klass
     * @returns {Function}
     *
     * @private
     */
    getCollectionDecoder: function(klass){
        return function(config){
            return (config && config.isInstance) ? config : new klass(config || {});
        };
    },
 
    /**
     * Returns a collection item key
     *
     * @param item
     * @return {String}
     *
     * @private
     */
    getCollectionItemKey: function(item){
        return item.getKey ? item.getKey() : (item._id || item.getId());
    }
});