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 countif(!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 serverreturn 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 handlervar _errorHandler = function(errorData, requestObject){var scope = keywordArgs.scope?keywordArgs.scope:dojo.global;if(keywordArgs.onError){keywordArgs.onError.call(scope, error);}};// Set up fetch handlervar _fetchHandler = function(items, requestObject){var scope = keywordArgs.scope?keywordArgs.scope:dojo.global;try{// There is supposed to be only one resultvar 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 hereif(keywordArgs.onItem){keywordArgs.onItem.call(scope, item);}}catch(error){if(keywordArgs.onError){keywordArgs.onError.call(scope, error);}}};// Construct queryvar request = {serverQuery:{id:keywordArgs.identity}};// Dispatch querythis._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, {});}