123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550 |
- 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 = allGetServerTime().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, {});
- }
|