Subversion Repositories Applications.papyrus

Rev

Blame | Last modification | View Log | RSS feed

if(!dojo._hasResource["dojox.data.QueryReadStore"]){ //_hasResource checks added by build. Do not use _hasResource directly in your code.
dojo._hasResource["dojox.data.QueryReadStore"] = true;
dojo.provide("dojox.data.QueryReadStore");
dojo.provide("dojox.data.QueryReadStore.InvalidItemError");
dojo.provide("dojox.data.QueryReadStore.InvalidAttributeError");

dojo.require("dojo.string");
dojo.require("dojo.data.util.simpleFetch");

dojo.declare("dojox.data.QueryReadStore", null, {
        /*
        //      summary:
        //              This class provides a store that is mainly intended to be used
        //              for loading data dynamically from the server, used i.e. for
        //              retreiving chunks of data from huge data stores on the server (by server-side filtering!).
        //              Upon calling the fetch() method of this store the data are requested from
        //              the server if they are not yet loaded for paging (or cached).
        //
        //              For example used for a combobox which works on lots of data. It
        //              can be used to retreive the data partially upon entering the
        //              letters "ac" it returns only items like "action", "acting", etc.
        //
        //  note:
        //      The field name "id" in a query is reserved for looking up data
        //      by id. This is necessary as before the first fetch, the store
        //      has no way of knowing which field the server will declare as
        //      identifier.
        //
        //      examples:
        //              // The parameter "query" contains the data that are sent to the server.
        //              var store = new dojox.data.QueryReadStore({url:'/search.php'});
        //              store.fetch({query:{name:'a'}, queryOptions:{ignoreCase:false}});
        //
        //              // Since "serverQuery" is given, it overrules and those data are
        //              // sent to the server.
        //              var store = new dojox.data.QueryReadStore({url:'/search.php'});
        //              store.fetch({serverQuery:{name:'a'}, queryOptions:{ignoreCase:false}});
        //
        //      todo:
        //              - there is a bug in the paging, when i set start:2, count:5 after an initial fetch() and doClientPaging:true
        //                it returns 6 elemetns, though count=5, try it in QueryReadStore.html
        //              - allow configuring if the paging shall takes place on the client or the server
        //              - add optional caching
        //              - when the first query searched for "a" and the next for a subset of
        //                the first, i.e. "ab" then we actually dont need a server request, if
        //                we have client paging, we just need to filter the items we already have
        //                that might also be tooo much logic
        */
        
        url:"",
        requestMethod:"get",
        //useCache:false,
        
        // We use the name in the errors, once the name is fixed hardcode it, may be.
        _className:"dojox.data.QueryReadStore",
        
        // This will contain the items we have loaded from the server.
        // The contents of this array is optimized to satisfy all read-api requirements
        // and for using lesser storage, so the keys and their content need some explaination:
        //              this._items[0].i - the item itself 
        //              this._items[0].r - a reference to the store, so we can identify the item
        //                      securly. We set this reference right after receiving the item from the
        //                      server.
        _items:[],
        
        // Store the last query that triggered xhr request to the server.
        // So we can compare if the request changed and if we shall reload 
        // (this also depends on other factors, such as is caching used, etc).
        _lastServerQuery:null,
        
        
        // Store a hash of the last server request. Actually I introduced this
        // for testing, so I can check if no unnecessary requests were issued for
        // client-side-paging.
        lastRequestHash:null,
        
        // If this is false, every request is sent to the server.
        // If it's true a second request with the same query will not issue another
        // request, but use the already returned data. This assumes that the server
        // does not do the paging.
        doClientPaging:true,

        // Items by identify for Identify API
        _itemsByIdentity:null,
        
        // Identifier used
        _identifier:null,

        _features: {'dojo.data.api.Read':true, 'dojo.data.api.Identity':true},
        
        constructor: function(/* Object */ params){
                dojo.mixin(this,params);
        },
        
        getValue: function(/* item */ item, /* attribute-name-string */ attribute, /* value? */ defaultValue){
                //      According to the Read API comments in getValue() and exception is
                //      thrown when an item is not an item or the attribute not a string!
                this._assertIsItem(item);
                if (!dojo.isString(attribute)) {
                        throw new Error(this._className+".getValue(): Invalid attribute, string expected!");
                }
                if(!this.hasAttribute(item, attribute)){
                        // read api says: return defaultValue "only if *item* does not have a value for *attribute*." 
                        // Is this the case here? The attribute doesn't exist, but a defaultValue, sounds reasonable.
                        if(defaultValue){
                                return defaultValue;
                        }
                        console.log(this._className+".getValue(): Item does not have the attribute '"+attribute+"'.");
                }
                return item.i[attribute];
        },
        
        getValues: function(/* item */ item, /* attribute-name-string */ attribute){
                var ret = [];
                if(this.hasAttribute(item, attribute)){
                        ret.push(item.i[attribute]);
                }
                return ret;
        },
        
        getAttributes: function(/* item */ item){
                this._assertIsItem(item);
                var ret = [];
                for(var i in item.i){
                        ret.push(i);
                }
                return ret;
        },

        hasAttribute: function(/* item */ item, /* attribute-name-string */ attribute) {
                //      summary: 
                //              See dojo.data.api.Read.hasAttribute()
                return this.isItem(item) && typeof item.i[attribute]!="undefined";
        },
        
        containsValue: function(/* item */ item, /* attribute-name-string */ attribute, /* anything */ value){
                var values = this.getValues(item, attribute);
                var len = values.length;
                for(var i=0; i<len; i++){
                        if(values[i]==value){
                                return true;
                        }
                }
                return false;
        },
        
        isItem: function(/* anything */ something){
                // Some basic tests, that are quick and easy to do here.
                // >>> var store = new dojox.data.QueryReadStore({});
                // >>> store.isItem("");
                // false
                //
                // >>> var store = new dojox.data.QueryReadStore({});
                // >>> store.isItem({});
                // false
                //
                // >>> var store = new dojox.data.QueryReadStore({});
                // >>> store.isItem(0);
                // false
                //
                // >>> var store = new dojox.data.QueryReadStore({});
                // >>> store.isItem({name:"me", label:"me too"});
                // false
                //
                if(something){
                        return typeof something.r!="undefined" && something.r==this;
                }
                return false;
        },
        
        isItemLoaded: function(/* anything */ something) {
                // Currently we dont have any state that tells if an item is loaded or not
                // if the item exists its also loaded.
                // This might change when we start working with refs inside items ...
                return this.isItem(something);
        },

        loadItem: function(/* object */ args){
                if(this.isItemLoaded(args.item)){
                        return;
                }
                // Actually we have nothing to do here, or at least I dont know what to do here ...
        },

        fetch:function(/* Object? */ request){
                //      summary:
                //              See dojo.data.util.simpleFetch.fetch() this is just a copy and I adjusted
                //              only the paging, since it happens on the server if doClientPaging is
                //              false, thx to http://trac.dojotoolkit.org/ticket/4761 reporting this.
                //              Would be nice to be able to use simpleFetch() to reduce copied code,
                //              but i dont know how yet. Ideas please!
                request = request || {};
                if(!request.store){
                        request.store = this;
                }
                var self = this;
        
                var _errorHandler = function(errorData, requestObject){
                        if(requestObject.onError){
                                var scope = requestObject.scope || dojo.global;
                                requestObject.onError.call(scope, errorData, requestObject);
                        }
                };
        
                var _fetchHandler = function(items, requestObject){
                        var oldAbortFunction = requestObject.abort || null;
                        var aborted = false;
                        
                        var startIndex = requestObject.start?requestObject.start:0;
                        if (self.doClientPaging==false) {
                                // For client paging we dont need no slicing of the result.
                                startIndex = 0;
                        }
                        var endIndex   = requestObject.count?(startIndex + requestObject.count):items.length;
        
                        requestObject.abort = function(){
                                aborted = true;
                                if(oldAbortFunction){
                                        oldAbortFunction.call(requestObject);
                                }
                        };
        
                        var scope = requestObject.scope || dojo.global;
                        if(!requestObject.store){
                                requestObject.store = self;
                        }
                        if(requestObject.onBegin){
                                requestObject.onBegin.call(scope, items.length, requestObject);
                        }
                        if(requestObject.sort){
                                items.sort(dojo.data.util.sorter.createSortFunction(requestObject.sort, self));
                        }
                        if(requestObject.onItem){
                                for(var i = startIndex; (i < items.length) && (i < endIndex); ++i){
                                        var item = items[i];
                                        if(!aborted){
                                                requestObject.onItem.call(scope, item, requestObject);
                                        }
                                }
                        }
                        if(requestObject.onComplete && !aborted){
                                var subset = null;
                                if (!requestObject.onItem) {
                                        subset = items.slice(startIndex, endIndex);
                                }
                                requestObject.onComplete.call(scope, subset, requestObject);   
                        }
                };
                this._fetchItems(request, _fetchHandler, _errorHandler);
                return request; // Object
        },

        getFeatures: function(){
                return this._features;
        },

        close: function(/*dojo.data.api.Request || keywordArgs || null */ request){
                // I have no idea if this is really needed ... 
        },

        getLabel: function(/* item */ item){
                // Override it to return whatever the label shall be, see Read-API.
                return undefined;
        },

        getLabelAttributes: function(/* item */ item){
                return null;
        },
        
        _fetchItems: function(request, fetchHandler, errorHandler){
                //      summary:
                //              The request contains the data as defined in the Read-API.
                //              Additionally there is following keyword "serverQuery".
                //
                //      The *serverQuery* parameter, optional.
                //              This parameter contains the data that will be sent to the server.
                //              If this parameter is not given the parameter "query"'s
                //              data are sent to the server. This is done for some reasons:
                //              - to specify explicitly which data are sent to the server, they
                //                might also be a mix of what is contained in "query", "queryOptions"
                //                and the paging parameters "start" and "count" or may be even
                //                completely different things.
                //              - don't modify the request.query data, so the interface using this
                //                store can rely on unmodified data, as the combobox dijit currently
                //                does it, it compares if the query has changed
                //              - request.query is required by the Read-API
                //
                //              I.e. the following examples might be sent via GET:
                //                fetch({query:{name:"abc"}, queryOptions:{ignoreCase:true}})
                //                the URL will become:   /url.php?name=abc
                //
                //                fetch({serverQuery:{q:"abc", c:true}, query:{name:"abc"}, queryOptions:{ignoreCase:true}})
                //                the URL will become:   /url.php?q=abc&c=true
                //                // The serverQuery-parameter has overruled the query-parameter
                //                // but the query parameter stays untouched, but is not sent to the server!
                //                // The serverQuery contains more data than the query, so they might differ!
                //

                var serverQuery = request.serverQuery || request.query || {};
                //Need to add start and count
                if(!this.doClientPaging){
                        serverQuery.start = request.start || 0;
                        // Count might not be sent if not given.
                        if (request.count) {
                                serverQuery.count = request.count;
                        }
                }
                // Compare the last query and the current query by simply json-encoding them,
                // so we dont have to do any deep object compare ... is there some dojo.areObjectsEqual()???
                if(this.doClientPaging && this._lastServerQuery!==null &&
                        dojo.toJson(serverQuery)==dojo.toJson(this._lastServerQuery)
                        ){
                        fetchHandler(this._items, request);
                }else{
                        var xhrFunc = this.requestMethod.toLowerCase()=="post" ? dojo.xhrPost : dojo.xhrGet;
                        var xhrHandler = xhrFunc({url:this.url, handleAs:"json-comment-optional", content:serverQuery});
                        xhrHandler.addCallback(dojo.hitch(this, function(data){
                                data=this._filterResponse(data);
                                this._items = [];
                                // Store a ref to "this" in each item, so we can simply check if an item
                                // really origins form here (idea is from ItemFileReadStore, I just don't know
                                // how efficient the real storage use, garbage collection effort, etc. is).
                                dojo.forEach(data.items,function(e){ 
                                        this._items.push({i:e, r:this}); 
                                },this); 
                                
                                var identifier = data.identifier;
                                this._itemsByIdentity = {};
                                if(identifier){
                                        this._identifier = identifier;
                                        for(i = 0; i < this._items.length; ++i){
                                                var item = this._items[i].i;
                                                var identity = item[identifier];
                                                if(!this._itemsByIdentity[identity]){
                                                        this._itemsByIdentity[identity] = item;
                                                }else{
                                                        throw new Error("dojo.data.QueryReadStore:  The json data as specified by: [" + this.url + "] is malformed.  Items within the list have identifier: [" + identifier + "].  Value collided: [" + identity + "]");
                                                }
                                        }
                                }else{
                                        this._identifier = Number;
                                        for(i = 0; i < this._items.length; ++i){
                                                this._items[i].n = i;
                                        }
                                }
                                
                                // TODO actually we should do the same as dojo.data.ItemFileReadStore._getItemsFromLoadedData() to sanitize
                                // (does it really sanititze them) and store the data optimal. should we? for security reasons???
                                fetchHandler(this._items, request);
                        }));
                        xhrHandler.addErrback(function(error){
                                errorHandler(error, request);
                        });
                        // Generate the hash using the time in milliseconds and a randon number.
                        // Since Math.randon() returns something like: 0.23453463, we just remove the "0."
                        // probably just for esthetic reasons :-).
                        this.lastRequestHash = new Date().getTime()+"-"+String(Math.random()).substring(2);
                        this._lastServerQuery = dojo.mixin({}, serverQuery);
                }
        },
        
        _filterResponse: function(data){
                //      summary:
                //              If the data from servers needs to be processed before it can be processed by this
                //              store, then this function should be re-implemented in subclass. This default 
                //              implementation just return the data unchanged.
                //      data:
                //              The data received from server
                return data;
        },

        _assertIsItem: function(/* item */ item){
                //      summary:
                //              It throws an error if item is not valid, so you can call it in every method that needs to
                //              throw an error when item is invalid.
                //      item: 
                //              The item to test for being contained by the store.
                if(!this.isItem(item)){
                        throw new dojox.data.QueryReadStore.InvalidItemError(this._className+": a function was passed an item argument that was not an item");
                }
        },

        _assertIsAttribute: function(/* attribute-name-string */ attribute){
                //      summary:
                //              This function tests whether the item passed in is indeed a valid 'attribute' like type for the store.
                //      attribute: 
                //              The attribute to test for being contained by the store.
                if(typeof attribute !== "string"){ 
                        throw new dojox.data.QueryReadStore.InvalidAttributeError(this._className+": '"+attribute+"' is not a valid attribute identifier.");
                }
        },

        fetchItemByIdentity: function(/* Object */ keywordArgs){
                //      summary: 
                //              See dojo.data.api.Identity.fetchItemByIdentity()

                // See if we have already loaded the item with that id
                // In case there hasn't been a fetch yet, _itemsByIdentity is null
                // and thus a fetch will be triggered below.
                if(this._itemsByIdentity){
                        var item = this._itemsByIdentity[keywordArgs.identity];
                        if(!(item === undefined)){
                                if(keywordArgs.onItem){
                                        var scope =  keywordArgs.scope?keywordArgs.scope:dojo.global;
                                        keywordArgs.onItem.call(scope, {i:item, r:this});
                                }
                                return;
                        }
                }

                // Otherwise we need to go remote
                // Set up error handler
                var _errorHandler = function(errorData, requestObject){
                        var scope =  keywordArgs.scope?keywordArgs.scope:dojo.global;
                        if(keywordArgs.onError){
                                keywordArgs.onError.call(scope, error);
                        }
                };
                
                // Set up fetch handler
                var _fetchHandler = function(items, requestObject){
                        var scope =  keywordArgs.scope?keywordArgs.scope:dojo.global;
                        try{
                                // There is supposed to be only one result
                                var item = null;
                                if(items && items.length == 1){
                                        item = items[0];
                                }
                                
                                // If no item was found, item is still null and we'll
                                // fire the onItem event with the null here
                                if(keywordArgs.onItem){
                                        keywordArgs.onItem.call(scope, item);
                                }
                        }catch(error){
                                if(keywordArgs.onError){
                                        keywordArgs.onError.call(scope, error);
                                }
                        }
                };
                
                // Construct query
                var request = {serverQuery:{id:keywordArgs.identity}};
                
                // Dispatch query
                this._fetchItems(request, _fetchHandler, _errorHandler);
        },
        
        getIdentity: function(/* item */ item){
                //      summary: 
                //              See dojo.data.api.Identity.getIdentity()
                var identifier = null;
                if(this._identifier === Number){
                        identifier = item.n; // Number
                }else{
                        identifier = item.i[this._identifier];
                }
                return identifier;
        },
        
        getIdentityAttributes: function(/* item */ item){
                //      summary:
                //              See dojo.data.api.Identity.getIdentityAttributes()
                return [this._identifier];
        }
});

dojo.declare("dojox.data.QueryReadStore.InvalidItemError", Error, {});
dojo.declare("dojox.data.QueryReadStore.InvalidAttributeError", Error, {});




}