// ========================================================================== // Project: SproutCore - JavaScript Application Framework // Copyright: ©2006-2009 Sprout Systems, Inc. and contributors. // Portions ©2008-2009 Apple Inc. All rights reserved. // License: Licensed under MIT license (see license.js) // ========================================================================== sc_require('core') ; sc_require('models/record'); /** @class This permits you to perform queries on your data store, written in a SQL-like language. Here is a simple example: {{{ q = SC.Query.create({ conditions: "firstName = 'Jonny' AND lastName = 'Cash'" }) }}} You can check if a certain record matches the query by calling: {{{ q.contains(record) }}} To find all records of your store, that match query q, use findAll with query q as argument: {{{ r = MyApp.store.findAll(q) }}} r will be a record array containing all matching records. To limit the query to a record type of MyApp.MyModel, you can specify the type as a property of the query like this: {{{ q = SC.Query.create({ conditions: "firstName = 'Jonny' AND lastName = 'Cash'", recordType: MyApp.MyModel }) }}} Calling find() like above will now return only records of type t. It is recommended to limit your query to a record type, since the query will have to look for matching records in the whole store, if no record type is given. You can give an order, which the resulting records should follow, like this: {{{ q = SC.Query.create({ conditions: "firstName = 'Jonny' AND lastName = 'Cash'", recordType: MyApp.MyModel, orderBy: "lastName, year DESC" }); }}} The default order direction is ascending. You can change it to descending by writing DESC behind the property name like in the example above. If no order is given, or records are equal in respect to a given order, records will be ordered by guid. h2. SproutCore Query Language Features of the query language: h4. Primitives: - record properties - null, undefined - true, false - numbers (integers and floats) - strings (double or single quoted) h4. Parameters: - %@ (wild card) - {parameterName} (named parameter) Wild cards are used to identify parameters by the order in which they appear in the query string. Named parameters can be used when tracking the order becomes difficult. Both types of parameters can be used by giving the parameters as a property to your query object: {{{ yourQuery.parameters = yourParameters }}} where yourParameters should have one of the following formats: for wild cards: [firstParam, secondParam, thirdParam] for named params: {name1: param1, mane2: parma2} You cannot use both types of parameters in a single query! h4. Operators: - = - != - < - <= - > - >= - BEGINS_WITH (checks if a string starts with another one) - ENDS_WITH (checks if a string ends with another one) - CONTAINS (checks if a string contains another one, or if an object is in an array) - MATCHES (checks if a string is matched by a regexp, you will have to use a parameter to insert the regexp) - ANY (checks if the thing on its left is contained in the array on its right, you will have to use a parameter to insert the array) - TYPE_IS (unary operator expecting a string containing the name of a Model class on its right side, only records of this type will match) h4. Boolean Operators: - AND - OR - NOT h4. Parenthesis for grouping: - ( and ) h2. Adding Your Own Query Handlers You can extend the query language with your own operators by calling: {{{ SC.Query.registerQueryExtension('your_operator', your_operator_definition); }}} See details below. As well you can provide your own comparison functions to control ordering of specific record properties like this: {{{ SC.Query.registerComparison(property_name, comparison_for_this_property); }}} h2. Examples Some example queries: TODO add examples @extends SC.Object @extends SC.Copyable @extends SC.Freezable @since SproutCore 1.0 */ SC.Query = SC.Object.extend(SC.Copyable, SC.Freezable, /** @scope SC.Query.prototype */ { // .......................................................... // PROPERTIES // /** Walk like a duck. @property {Boolean} */ isQuery: YES, /** Unparsed query conditions. If you are handling a query yourself, then you will find the base query string here. @property {String} */ conditions: null, /** Optional orderBy parameters. This can be a string of keys, optionally beginning with the strings "DESC " or "ASC " to select descending or ascending order. Alternatively, you can specify a comparison function, in which case the two records will be sent to it. Your comparison function, as with any other, is expected to return -1, 0, or 1. @property {String | Function} */ orderBy: null, /** The base record type or types for the query. This must be specified to filter the kinds of records this query will work on. You may either set this to a single record type or to an array or set of record types. @property {SC.Record} */ recordType: null, /** Optional array of multiple record types. If the query accepts multiple record types, this is how you can check for it. @property {SC.Enumerable} */ recordTypes: null, /** Returns the complete set of recordTypes matched by this query. Includes any named recordTypes plus their subclasses. @property {SC.Enumerable} */ expandedRecordTypes: function() { var ret = SC.CoreSet.create(), rt, q ; if (rt = this.get('recordType')) this._scq_expandRecordType(rt, ret); else if (rt = this.get('recordTypes')) { rt.forEach(function(t) { this._scq_expandRecordType(t, ret); }, this); } else this._scq_expandRecordType(SC.Record, ret); // save in queue. if a new recordtype is defined, we will be notified. q = SC.Query._scq_queriesWithExpandedRecordTypes; if (!q) { q = SC.Query._scq_queriesWithExpandedRecordTypes = SC.CoreSet.create(); } q.add(this); return ret.freeze() ; }.property('recordType', 'recordTypes').cacheable(), /** @private expands a single record type into the set. called recursively */ _scq_expandRecordType: function(recordType, set) { if (set.contains(recordType)) return; // nothing to do set.add(recordType); if (SC.typeOf(recordType)===SC.T_STRING) { recordType = SC.objectForPropertyPath(recordType); } recordType.subclasses.forEach(function(t) { this._scq_expandRecordType(t, set); }, this); }, /** Optional hash of parameters. These parameters may be interpolated into the query conditions. If you are handling the query manually, these parameters will not be used. @property {Hash} */ parameters: null, /** Indicates the location where the result set for this query is stored. Currently the available options are: - SC.Query.LOCAL: indicates that the query results will be automatically computed from the in-memory store. - SC.Query.REMOTE: indicates that the query results are kept on a remote server and hence must be loaded from the DataSource. The default setting for this property is SC.Query.LOCAL. Note that even if a query location is LOCAL, your DataSource will still have its fetch() method called for the query. For LOCAL queries, you won't need to explicitly provide the query result set; you can just load records into the in-memory store as needed and let the query recompute automatically. If your query location is REMOTE, then your DataSource will need to provide the actual set of query results manually. Usually you will only need to use a REMOTE query if you are retrieving a large data set and you don't want to pay the cost of computing the result set client side. @property {String} */ location: 'local', // SC.Query.LOCAL /** Another query that will optionally limit the search of records. This is usually configured for you when you do find() from another record array. @property {SC.Query} */ scope: null, /** Returns YES if query location is Remote. This is sometimes more convenient than checking the location. @property {Boolean} */ isRemote: function() { return this.get('location') === SC.Query.REMOTE; }.property('location').cacheable(), /** Returns YES if query location is Local. This is sometimes more convenient than checking the location. @property {Boolean} */ isLocal: function() { return this.get('location') === SC.Query.LOCAL; }.property('location').cacheable(), /** Indicates whether a record is editable or not. Defaults to NO. Local queries should never be made editable. Remote queries may be editable or not depending on the data source. */ isEditable: NO, // .......................................................... // PRIMITIVE METHODS // /** Returns YES if record is matched by the query, NO otherwise. This is used when computing a query locally. @param {SC.Record} record the record to check @param {Hash} parameters optional override parameters @returns {Boolean} YES if record belongs, NO otherwise */ contains: function(record, parameters) { // check the recordType if specified var rtype, ret = YES ; if (rtype = this.get('recordTypes')) { // plural form ret = rtype.find(function(t) { return SC.kindOf(record, t); }); } else if (rtype = this.get('recordType')) { // singular ret = SC.kindOf(record, rtype); } if (!ret) return NO ; // if either did not pass, does not contain // if we have a scope - check for that as well var scope = this.get('scope'); if (scope && !scope.contains(record)) return NO ; // now try parsing if (!this._isReady) this.parse(); // prepare the query if needed if (!this._isReady) return NO ; if (parameters === undefined) parameters = this.parameters || this; // if parsing worked we check if record is contained // if parsing failed no record will be contained return this._tokenTree.evaluate(record, parameters); }, /** Returns YES if the query matches one or more of the record types in the passed set. @param {SC.Set} types set of record types @returns {Boolean} YES if record types match */ containsRecordTypes: function(types) { var rtype = this.get('recordType'); if (rtype) { return !!types.find(function(t) { return SC.kindOf(t, rtype); }); } else if (rtype = this.get('recordTypes')) { return !!rtype.find(function(t) { return !!types.find(function(t2) { return SC.kindOf(t2,t); }); }); } else return YES; // allow anything through }, /** Returns the sort order of the two passed records, taking into account the orderBy property set on this query. This method does not verify that the two records actually belong in the query set or not; this is checked using contains(). @param {SC.Record} record1 the first record @param {SC.Record} record2 the second record @returns {Number} -1 if record1 < record2, +1 if record1 > record2, 0 if equal */ compare: function(record1, record2) { // IMPORTANT: THIS CODE IS ALSO INLINED INSIDE OF THE 'compareStoreKeys' // CLASS METHOD. IF YOU CHANGE THIS IMPLEMENTATION, BE SURE // TO UPDATE IT THERE, TOO. // // (Any clients overriding this method will have their version called, // however. That's why we'll keep this here; clients might want to // override it and call sc_super()). var result = 0, propertyName, order, len, i; // fast cases go here if (record1 === record2) return 0; // if called for the first time we have to build the order array if (!this._isReady) this.parse(); if (!this._isReady) { // can't parse. guid is wrong but consistent return SC.compare(record1.get('id'),record2.get('id')); } // For every property specified in orderBy until non-eql result is found. // Or, if orderBy is a comparison function, simply invoke it with the // records. order = this._order; if (SC.typeOf(order) === SC.T_FUNCTION) { result = order.call(null, record1, record2); } else { len = order ? order.length : 0; for (i=0; result===0 && (i < len); i++) { propertyName = order[i].propertyName; // if this property has a registered comparison use that if (SC.Query.comparisons[propertyName]) { result = SC.Query.comparisons[propertyName]( record1.get(propertyName),record2.get(propertyName)); // if not use default SC.compare() } else { result = SC.compare( record1.get(propertyName), record2.get(propertyName) ); } if ((result!==0) && order[i].descending) result = (-1) * result; } } // return result or compare by guid if (result !== 0) return result ; else return SC.compare(record1.get('id'),record2.get('id')); }, /** @private Becomes YES once the query has been successfully parsed */ _isReady: NO, /** This method has to be called before the query object can be used. You will normaly not have to do this, it will be called automatically if you try to evaluate a query. You can however use this function for testing your queries. @returns {Boolean} true if parsing succeeded, false otherwise */ parse: function() { var conditions = this.get('conditions'), lang = this.get('queryLanguage'), tokens, tree; tokens = this._tokenList = this.tokenizeString(conditions, lang); tree = this._tokenTree = this.buildTokenTree(tokens, lang); this._order = this.buildOrder(this.get('orderBy')); this._isReady = !!tree && !tree.error; if (tree && tree.error) throw tree.error; return this._isReady; }, /** Returns the same query but with the scope set to the passed record array. This will copy the receiver. It also stores these queries in a cache to reuse them if possible. @param {SC.RecordArray} recordArray the scope @returns {SC.Query} new query */ queryWithScope: function(recordArray) { // look for a cached query on record array. var key = SC.keyFor('__query__', SC.guidFor(this)), ret = recordArray[key]; if (!ret) { recordArray[key] = ret = this.copy(); ret.set('scope', recordArray); ret.freeze(); } return ret ; }, // .......................................................... // PRIVATE SUPPORT // /** @private Properties that need to be copied when cloning the query. */ copyKeys: 'conditions orderBy recordType recordTypes parameters location scope'.w(), /** @private */ concatenatedProperties: 'copyKeys'.w(), /** @private Implement the Copyable API to clone a query object once it has been created. */ copy: function() { var opts = {}, keys = this.get('copyKeys'), loc = keys ? keys.length : 0, key, value, ret; while(--loc >= 0) { key = keys[loc]; value = this.get(key); if (value !== undefined) opts[key] = value ; } ret = this.constructor.create(opts); opts = null; return ret ; }, // .......................................................... // QUERY LANGUAGE DEFINITION // /** This is the definition of the query language. You can extend it by using SC.Query.registerQueryExtension(). */ queryLanguage: { 'UNKNOWN': { firstCharacter: /[^\s'"\w\d\(\)\{\}]/, notAllowed: /[\s'"\w\d\(\)\{\}]/ }, 'PROPERTY': { firstCharacter: /[a-zA-Z_]/, notAllowed: /[^a-zA-Z_0-9]/, evalType: 'PRIMITIVE', /** @ignore */ evaluate: function (r,w) { return r.get(this.tokenValue); } }, 'NUMBER': { firstCharacter: /[\d\-]/, notAllowed: /[^\d\-\.]/, format: /^-?\d+$|^-?\d+\.\d+$/, evalType: 'PRIMITIVE', /** @ignore */ evaluate: function (r,w) { return parseFloat(this.tokenValue); } }, 'STRING': { firstCharacter: /['"]/, delimeted: true, evalType: 'PRIMITIVE', /** @ignore */ evaluate: function (r,w) { return this.tokenValue; } }, 'PARAMETER': { firstCharacter: /\{/, lastCharacter: '}', delimeted: true, evalType: 'PRIMITIVE', /** @ignore */ evaluate: function (r,w) { return w[this.tokenValue]; } }, '%@': { rememberCount: true, reservedWord: true, evalType: 'PRIMITIVE', /** @ignore */ evaluate: function (r,w) { return w[this.tokenValue]; } }, 'OPEN_PAREN': { firstCharacter: /\(/, singleCharacter: true }, 'CLOSE_PAREN': { firstCharacter: /\)/, singleCharacter: true }, 'AND': { reservedWord: true, leftType: 'BOOLEAN', rightType: 'BOOLEAN', evalType: 'BOOLEAN', /** @ignore */ evaluate: function (r,w) { var left = this.leftSide.evaluate(r,w); var right = this.rightSide.evaluate(r,w); return left && right; } }, 'OR': { reservedWord: true, leftType: 'BOOLEAN', rightType: 'BOOLEAN', evalType: 'BOOLEAN', /** @ignore */ evaluate: function (r,w) { var left = this.leftSide.evaluate(r,w); var right = this.rightSide.evaluate(r,w); return left || right; } }, 'NOT': { reservedWord: true, rightType: 'BOOLEAN', evalType: 'BOOLEAN', /** @ignore */ evaluate: function (r,w) { var right = this.rightSide.evaluate(r,w); return !right; } }, '=': { reservedWord: true, leftType: 'PRIMITIVE', rightType: 'PRIMITIVE', evalType: 'BOOLEAN', /** @ignore */ evaluate: function (r,w) { var left = this.leftSide.evaluate(r,w); var right = this.rightSide.evaluate(r,w); return SC.isEqual(left, right); } }, '!=': { reservedWord: true, leftType: 'PRIMITIVE', rightType: 'PRIMITIVE', evalType: 'BOOLEAN', /** @ignore */ evaluate: function (r,w) { var left = this.leftSide.evaluate(r,w); var right = this.rightSide.evaluate(r,w); return !SC.isEqual(left, right); } }, '<': { reservedWord: true, leftType: 'PRIMITIVE', rightType: 'PRIMITIVE', evalType: 'BOOLEAN', /** @ignore */ evaluate: function (r,w) { var left = this.leftSide.evaluate(r,w); var right = this.rightSide.evaluate(r,w); return SC.compare(left, right) == -1; //left < right; } }, '<=': { reservedWord: true, leftType: 'PRIMITIVE', rightType: 'PRIMITIVE', evalType: 'BOOLEAN', /** @ignore */ evaluate: function (r,w) { var left = this.leftSide.evaluate(r,w); var right = this.rightSide.evaluate(r,w); return SC.compare(left, right) != 1; //left <= right; } }, '>': { reservedWord: true, leftType: 'PRIMITIVE', rightType: 'PRIMITIVE', evalType: 'BOOLEAN', /** @ignore */ evaluate: function (r,w) { var left = this.leftSide.evaluate(r,w); var right = this.rightSide.evaluate(r,w); return SC.compare(left, right) == 1; //left > right; } }, '>=': { reservedWord: true, leftType: 'PRIMITIVE', rightType: 'PRIMITIVE', evalType: 'BOOLEAN', /** @ignore */ evaluate: function (r,w) { var left = this.leftSide.evaluate(r,w); var right = this.rightSide.evaluate(r,w); return SC.compare(left, right) != -1; //left >= right; } }, 'BEGINS_WITH': { reservedWord: true, leftType: 'PRIMITIVE', rightType: 'PRIMITIVE', evalType: 'BOOLEAN', /** @ignore */ evaluate: function (r,w) { var all = this.leftSide.evaluate(r,w); var start = this.rightSide.evaluate(r,w); return ( all && all.indexOf(start) === 0 ); } }, 'ENDS_WITH': { reservedWord: true, leftType: 'PRIMITIVE', rightType: 'PRIMITIVE', evalType: 'BOOLEAN', /** @ignore */ evaluate: function (r,w) { var all = this.leftSide.evaluate(r,w); var end = this.rightSide.evaluate(r,w); return ( all && all.indexOf(end) === (all.length - end.length) ); } }, 'CONTAINS': { reservedWord: true, leftType: 'PRIMITIVE', rightType: 'PRIMITIVE', evalType: 'BOOLEAN', /** @ignore */ evaluate: function (r,w) { var all = this.leftSide.evaluate(r,w) || []; var value = this.rightSide.evaluate(r,w); switch(SC.typeOf(all)) { case SC.T_STRING: return (all.indexOf(value) !== -1); case SC.T_ARRAY: var found = false; var i = 0; while ( found===false && i 0 ) return true; else return false; } function tokenIsMissingChilds (position) { var p = position; if ( p < 0 ) return true; return (expectedType('left',p) && !l[p].leftSide) || (expectedType('right',p) && !l[p].rightSide); } function typesAreMatching (parent, child) { var side = (child < parent) ? 'left' : 'right'; if ( parent < 0 || child < 0 ) return false; if ( !expectedType(side,parent) ) return false; if ( !evalType(child) ) return false; if ( expectedType(side,parent) == evalType(child) ) return true; else return false; } function preceedingTokenCanBeMadeChild (position) { var p = position; if ( !tokenIsMissingChilds(p) ) return false; if ( !preceedingTokenExists(p) ) return false; if ( typesAreMatching(p,p-1) ) return true; else return false; } function preceedingTokenCanBeMadeParent (position) { var p = position; if ( tokenIsMissingChilds(p) ) return false; if ( !preceedingTokenExists(p) ) return false; if ( !tokenIsMissingChilds(p-1) ) return false; if ( typesAreMatching(p-1,p) ) return true; else return false; } function makeChild (position) { var p = position; if (p<1) return false; l[p].leftSide = l[p-1]; removeToken(p-1); } function makeParent (position) { var p = position; if (p<1) return false; l[p-1].rightSide = l[p]; removeToken(p); } function removeParenthesesPair (position) { removeToken(position); removeToken(openParenthesisStack.pop()); } // step through the tokenList for (i=0; i < l.length; i++) { shouldCheckAgain = false; if ( l[i].tokenType == 'UNKNOWN' ) { error.push('found unknown token: '+l[i].tokenValue); } if ( l[i].tokenType == 'OPEN_PAREN' ) openParenthesisStack.push(i); if ( l[i].tokenType == 'CLOSE_PAREN' ) removeParenthesesPair(i); if ( preceedingTokenCanBeMadeChild(i) ) makeChild(i); if ( preceedingTokenCanBeMadeParent(i) ){ makeParent(i); shouldCheckAgain = true; } if ( shouldCheckAgain ) i--; } // error if tokenList l is not a single token now if (l.length == 1) l = l[0]; else error.push('string did not resolve to a single tree'); // error? if (error.length > 0) return {error: error.join(',\n'), tree: l}; // everything fine - token list is now a tree and can be returned else return l; }, // .......................................................... // ORDERING // /** Takes a string containing an order statement and returns an array describing this order for easier processing. Called by parse(). @param {String | Function} orderOp the string containing the order statement, or a comparison function @returns {Array | Function} array of order statement, or a function if a function was specified */ buildOrder: function (orderOp) { if (!orderOp) { return []; } else if (SC.typeOf(orderOp) === SC.T_FUNCTION) { return orderOp; } else { var o = orderOp.split(','); for (var i=0; i < o.length; i++) { var p = o[i]; p = p.replace(/^\s+|\s+$/,''); p = p.replace(/\s+/,','); p = p.split(','); o[i] = {propertyName: p[0]}; if (p[1] && p[1] == 'DESC') o[i].descending = true; } return o; } } }); // Class Methods SC.Query.mixin( /** @scope SC.Query */ { /** Constant used for SC.Query#location @property {String} */ LOCAL: 'local', /** Constant used for SC.Query#location @property {String} */ REMOTE: 'remote', /** Given a query, returns the associated storeKey. For the inverse of this method see SC.Store.queryFor(). @param {SC.Query} query the query @returns {Number} a storeKey. */ storeKeyFor: function(query) { return query ? query.get('storeKey') : null; }, /** Will find which records match a give SC.Query and return an array of store keys. This will also apply the sorting for the query. @param {SC.Query} query to apply @param {SC.RecordArray} records to search within @param {SC.Store} store to materialize record from @returns {Array} array instance of store keys matching the SC.Query (sorted) */ containsRecords: function(query, records, store) { var ret = []; for(var idx=0,len=records.get('length');idx record2, 0 if equal */ compareStoreKeys: function(storeKey1, storeKey2) { var K = SC.Query, tempStores = K._TMP_STORES, tempQueries = K._TMP_QUERIES, store = tempStores[tempStores.length - 1], query = tempQueries[tempQueries.length - 1], compareFunc = query.compare, record1 = store.materializeRecord(storeKey1), record2 = store.materializeRecord(storeKey2); // If the query implements a custom 'compare' function, then use it. // Otherwise, we have the logic from the standard version inlined here. if (compareFunc !== K.prototype.compare) { return compareFunc.call(query, record1, record2); } else { // THIS CODE IS THE SAME AS THE 'compare' METHOD, EXCEPT THAT 'this' HAS // BEEN CHANGED TO 'query'. // // It is inlined here to avoid the extra method invocation in the // typical case where the client does not supply a custom 'compare' // function. var result = 0, propertyName, order, len, i; // fast cases go here if (record1 === record2) return 0; // if called for the first time we have to build the order array if (!query._isReady) query.parse(); if (!query._isReady) { // can't parse. guid is wrong but consistent return SC.compare(record1.get('id'),record2.get('id')); } // For every property specified in orderBy until non-eql result is found. // Or, if orderBy is a comparison function, simply invoke it with the // records. order = query._order; if (SC.typeOf(order) === SC.T_FUNCTION) { result = order.call(null, record1, record2); } else { len = order ? order.length : 0; for (i=0; result===0 && (i < len); i++) { propertyName = order[i].propertyName; // if query property has a registered comparison use that if (SC.Query.comparisons[propertyName]) { result = SC.Query.comparisons[propertyName]( record1.get(propertyName),record2.get(propertyName)); // if not use default SC.compare() } else { result = SC.compare( record1.get(propertyName), record2.get(propertyName) ); } if ((result!==0) && order[i].descending) result = (-1) * result; } } // return result or compare by guid if (result !== 0) return result ; else return SC.compare(record1.get('id'),record2.get('id')); } }, /** Returns a SC.Query instance reflecting the passed properties. Where possible this method will return cached query instances so that multiple calls to this method will return the same instance. This is not possible however, when you pass custom parameters or set ordering. All returned queries are frozen. Usually you will not call this method directly. Instead use the more convenient SC.Query.local() and SC.Query.remote(). h2. Examples There are a number of different ways you can call this method. The following return local queries selecting all records of a particular type or types, including any subclasses: {{{ var people = SC.Query.local(Ab.Person); var peopleAndCompanies = SC.Query.local([Ab.Person, Ab.Company]); var people = SC.Query.local('Ab.Person'); var peopleAndCompanies = SC.Query.local('Ab.Person Ab.Company'.w()); var allRecords = SC.Query.local(SC.Record); }}} The following will match a particular type of condition: {{{ var married = SC.Query.local(Ab.Person, "isMarried=YES"); var married = SC.Query.local(Ab.Person, "isMarried=%@", [YES]); var married = SC.Query.local(Ab.Person, "isMarried={married}", { married: YES }); }}} You can also pass a hash of options as the second parameter. This is how you specify an order, for example: {{{ var orderedPeople = SC.Query.local(Ab.Person, { orderBy: "firstName" }); }}} @param {String} location the query location. @param {SC.Record|Array} recordType the record type or types. @param {String} conditions optional conditions @param {Hash} params optional params. or pass multiple args. @returns {SC.Query} */ build: function(location, recordType, conditions, params) { var opts = null, ret, cache, key, tmp; // fast case for query objects. if (recordType && recordType.isQuery) { if (recordType.get('location') === location) return recordType; else return recordType.copy().set('location', location).freeze(); } // normalize recordType if (typeof recordType === SC.T_STRING) { ret = SC.objectForPropertyPath(recordType); if (!ret) throw "%@ did not resolve to a class".fmt(recordType); recordType = ret ; } else if (recordType && recordType.isEnumerable) { ret = []; recordType.forEach(function(t) { if (typeof t === SC.T_STRING) t = SC.objectForPropertyPath(t); if (!t) throw "cannot resolve record types: %@".fmt(recordType); ret.push(t); }, this); recordType = ret ; } else if (!recordType) recordType = SC.Record; // find all records if (params === undefined) params = null; if (conditions === undefined) conditions = null; // normalize other params. if conditions is just a hash, treat as opts if (!params && (typeof conditions !== SC.T_STRING)) { opts = conditions; conditions = null ; } // special case - easy to cache. if (!params && !opts) { tmp = SC.Query._scq_recordTypeCache; if (!tmp) tmp = SC.Query._scq_recordTypeCache = {}; cache = tmp[location]; if (!cache) cache = tmp[location] = {}; if (recordType.isEnumerable) { key = recordType.map(function(k) { return SC.guidFor(k); }); key = key.sort().join(':'); } else key = SC.guidFor(recordType); if (conditions) key = [key, conditions].join('::'); ret = cache[key]; if (!ret) { if (recordType.isEnumerable) { opts = { recordTypes: recordType.copy() }; } else opts = { recordType: recordType }; opts.location = location ; opts.conditions = conditions ; ret = cache[key] = SC.Query.create(opts).freeze(); } // otherwise parse extra conditions and handle them } else { if (!opts) opts = {}; if (!opts.location) opts.location = location ; // allow override // pass one or more recordTypes. if (recordType && recordType.isEnumerable) { opts.recordsTypes = recordType; } else opts.recordType = recordType; // set conditions and params if needed if (conditions) opts.conditions = conditions; if (params) opts.parameters = params; ret = SC.Query.create(opts).freeze(); } return ret ; }, /** Returns a LOCAL query with the passed options. For a full description of the parameters you can pass to this method, see SC.Query.build(). @param {SC.Record|Array} recordType the record type or types. @param {String} conditions optional conditions @param {Hash} params optional params. or pass multiple args. @returns {SC.Query} */ local: function(recordType, conditions, params) { return this.build(SC.Query.LOCAL, recordType, conditions, params); }, /** Returns a REMOTE query with the passed options. For a full description of the parameters you can pass to this method, see SC.Query.build(). @param {SC.Record|Array} recordType the record type or types. @param {String} conditions optional conditions @param {Hash} params optional params. or pass multiple args. @returns {SC.Query} */ remote: function(recordType, conditions, params) { return this.build(SC.Query.REMOTE, recordType, conditions, params); }, /** @private called by SC.Record.extend(). invalided expandedRecordTypes */ _scq_didDefineRecordType: function() { var q = SC.Query._scq_queriesWithExpandedRecordTypes; if (q) { q.forEach(function(query) { query.notifyPropertyChange('expandedRecordTypes'); }, this); q.clear(); } } }); /** @private Hash of registered comparisons by propery name. */ SC.Query.comparisons = {}; /** Call to register a comparison for a specific property name. The function you pass should accept two values of this property and return -1 if the first is smaller than the second, 0 if they are equal and 1 if the first is greater than the second. @param {String} name of the record property @param {Function} custom comparison function @returns {SC.Query} receiver */ SC.Query.registerComparison = function(propertyName, comparison) { SC.Query.comparisons[propertyName] = comparison; }; /** Call to register an extension for the query language. You shoud provide a name for your extension and a definition specifying how it should be parsed and evaluated. Have a look at queryLanguage for examples of definitions. TODO add better documentation here @param {String} tokenName name of the operator @param {Object} token extension definition @returns {SC.Query} receiver */ SC.Query.registerQueryExtension = function(tokenName, token) { SC.Query.prototype.queryLanguage[tokenName] = token; }; // shorthand SC.Q = SC.Query.from ;