/**
 * 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,
        
        /**
         * @cfg {String[]} autoGenerateKey
         *
         * Set an array of config names available on this class that will
         * be used to generate a key. This key is useful when instances
         * of this class are used in a {@link Ext.util.Collection}.
         *
         * The key is generated by combining the values of those configs.
         * If a config value is an instance of a class then it will look
         * for a getKey function on that class and use it.
         *
         * Set to false if you don't want to generate they key
         */
        autoGenerateKey: {
            $value: false,
            merge: function(newValue, oldValue) {
                return newValue ? Ext.concat(newValue, oldValue || null) : false;
            }
        }
    },
 
    // 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.prototype.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());
    },
 
    getKey: function() {
        var generate = this.getAutoGenerateKey(),
            key = '',
            config = this.getConfig(),
            len, i;
 
        if (!Ext.isArray(generate) || !generate.length) {
            return this.getId();
        }
 
        len = generate.length;
 
        for (= 0; i < len; i++) {
            key += this.serializeKeyValue(config[generate[i]]);
        }
 
        return key;
    },
 
    serializeKeyValue: function(value) {
        var key = '',
            keys, len, i;
 
        if (value === null) {
            key = '_';
        }
        else if (Ext.isDate(value)) {
            key = value.getTime();
        }
        else if (Ext.isArray(value)) {
            len = value.length;
 
            for (= 0; i < len; i++) {
                key += this.serializeKeyValue(value[i]);
            }
        }
        else if (typeof value === 'object') {
            if (value.isInstance) {
                key = value.getKey ? value.getKey() : '_';
            }
            else {
                keys = Ext.Object.getAllKeys(value);
                keys = Ext.Array.sort(keys);
                len = keys.length;
 
                for (= 0; i < len; i++) {
                    key += this.serializeKeyValue(value[keys[i]]);
                }
            }
        }
        else {
            key = value;
        }
 
        return key;
    }
});