This function can e.g. be used to get a symbolizer literal with an ever increasing zIndex.
970 lines
31 KiB
JavaScript
970 lines
31 KiB
JavaScript
goog.provide('ol.test.expression');
|
|
|
|
|
|
describe('ol.expr.parse()', function() {
|
|
|
|
it('parses a subset of ECMAScript 5.1 expressions', function() {
|
|
var expr = ol.expr.parse('foo');
|
|
expect(expr).to.be.a(ol.expr.Expression);
|
|
});
|
|
|
|
describe('11.1 - primary expressions', function() {
|
|
// http://www.ecma-international.org/ecma-262/5.1/#sec-11.1
|
|
|
|
it('parses identifier expressions', function() {
|
|
var expr = ol.expr.parse('foo');
|
|
expect(expr).to.be.a(ol.expr.Identifier);
|
|
expect(expr.evaluate({foo: 'bar'})).to.be('bar');
|
|
});
|
|
|
|
it('consumes whitespace as expected', function() {
|
|
var expr = ol.expr.parse(' foo ');
|
|
expect(expr).to.be.a(ol.expr.Identifier);
|
|
expect(expr.evaluate({foo: 'bar'})).to.be('bar');
|
|
});
|
|
|
|
it('throws on invalid identifier expressions', function() {
|
|
expect(function() {
|
|
ol.expr.parse('3foo');
|
|
}).throwException(function(err) {
|
|
expect(err).to.be.an(ol.expr.UnexpectedToken);
|
|
var token = err.token;
|
|
expect(token.value).to.be('f');
|
|
expect(token.index).to.be(1);
|
|
});
|
|
});
|
|
|
|
it('parses string literal expressions', function() {
|
|
var expr = ol.expr.parse('"foo"');
|
|
expect(expr).to.be.a(ol.expr.Literal);
|
|
expect(expr.evaluate()).to.be('foo');
|
|
});
|
|
|
|
it('throws on unterminated string', function() {
|
|
expect(function() {
|
|
ol.expr.parse('"foo');
|
|
}).throwException(function(err) {
|
|
expect(err).to.be.an(ol.expr.UnexpectedToken);
|
|
var token = err.token;
|
|
expect(token.type).to.be(ol.expr.TokenType.EOF);
|
|
expect(token.index).to.be(4);
|
|
});
|
|
});
|
|
|
|
it('parses numeric literal expressions', function() {
|
|
var expr = ol.expr.parse('.42e+2');
|
|
expect(expr).to.be.a(ol.expr.Literal);
|
|
expect(expr.evaluate()).to.be(42);
|
|
});
|
|
|
|
it('throws on invalid number', function() {
|
|
expect(function() {
|
|
ol.expr.parse('.42eX');
|
|
}).throwException(function(err) {
|
|
expect(err).to.be.an(ol.expr.UnexpectedToken);
|
|
var token = err.token;
|
|
expect(token.value).to.be('X');
|
|
expect(token.index).to.be(4);
|
|
});
|
|
});
|
|
|
|
it('parses boolean literal expressions', function() {
|
|
var expr = ol.expr.parse('false');
|
|
expect(expr).to.be.a(ol.expr.Literal);
|
|
expect(expr.evaluate()).to.be(false);
|
|
});
|
|
|
|
it('parses null literal expressions', function() {
|
|
var expr = ol.expr.parse('null');
|
|
expect(expr).to.be.a(ol.expr.Literal);
|
|
expect(expr.evaluate()).to.be(null);
|
|
});
|
|
|
|
});
|
|
|
|
describe('11.2 - left-hand-side expressions', function() {
|
|
// http://www.ecma-international.org/ecma-262/5.1/#sec-11.2
|
|
|
|
it('parses member expressions with dot notation', function() {
|
|
var expr = ol.expr.parse('foo.bar.baz');
|
|
expect(expr).to.be.a(ol.expr.Member);
|
|
var scope = {foo: {bar: {baz: 42}}};
|
|
expect(expr.evaluate(scope)).to.be(42);
|
|
});
|
|
|
|
it('consumes whitespace as expected', function() {
|
|
var expr = ol.expr.parse(' foo . bar . baz ');
|
|
expect(expr).to.be.a(ol.expr.Member);
|
|
var scope = {foo: {bar: {baz: 42}}};
|
|
expect(expr.evaluate(scope)).to.be(42);
|
|
});
|
|
|
|
it('throws on invalid member expression', function() {
|
|
expect(function() {
|
|
ol.expr.parse('foo.4bar');
|
|
}).throwException(function(err) {
|
|
expect(err).to.be.an(ol.expr.UnexpectedToken);
|
|
var token = err.token;
|
|
expect(token.value).to.be('b');
|
|
expect(token.index).to.be(5);
|
|
});
|
|
});
|
|
|
|
it('parses call expressions with literal arguments', function() {
|
|
var expr = ol.expr.parse('foo(42, "bar")');
|
|
expect(expr).to.be.a(ol.expr.Call);
|
|
var scope = {
|
|
foo: function(num, str) {
|
|
expect(num).to.be(42);
|
|
expect(str).to.be('bar');
|
|
return str + num;
|
|
}
|
|
};
|
|
expect(expr.evaluate(scope)).to.be('bar42');
|
|
});
|
|
|
|
it('throws on calls with unterminated arguments', function() {
|
|
expect(function() {
|
|
ol.expr.parse('foo(42,)');
|
|
}).throwException(function(err) {
|
|
expect(err).to.be.an(ol.expr.UnexpectedToken);
|
|
var token = err.token;
|
|
expect(token.value).to.be(')');
|
|
expect(token.index).to.be(7);
|
|
});
|
|
});
|
|
|
|
});
|
|
|
|
describe('11.3 - postfix expressions', function() {
|
|
// http://www.ecma-international.org/ecma-262/5.1/#sec-11.3
|
|
it('not supported');
|
|
});
|
|
|
|
|
|
describe('11.4 - unary operators', function() {
|
|
// http://www.ecma-international.org/ecma-262/5.1/#sec-11.4
|
|
|
|
it('parses logical not operator', function() {
|
|
var expr = ol.expr.parse('!foo');
|
|
expect(expr).to.be.a(ol.expr.Not);
|
|
expect(expr.evaluate({foo: true})).to.be(false);
|
|
expect(expr.evaluate({foo: false})).to.be(true);
|
|
expect(expr.evaluate({foo: ''})).to.be(true);
|
|
expect(expr.evaluate({foo: 'foo'})).to.be(false);
|
|
});
|
|
|
|
it('consumes whitespace as expected', function() {
|
|
var expr = ol.expr.parse(' ! foo');
|
|
expect(expr).to.be.a(ol.expr.Not);
|
|
expect(expr.evaluate({foo: true})).to.be(false);
|
|
expect(expr.evaluate({foo: false})).to.be(true);
|
|
});
|
|
|
|
it('parses not preceeding call expression', function() {
|
|
var lib = {
|
|
foo: function() {
|
|
return true;
|
|
}
|
|
};
|
|
var expr = ol.expr.parse('!foo()');
|
|
expect(expr).to.be.a(ol.expr.Not);
|
|
expect(expr.evaluate(null, lib)).to.be(false);
|
|
});
|
|
|
|
it('parses not in call argument', function() {
|
|
var lib = {
|
|
foo: function(arg) {
|
|
return arg;
|
|
}
|
|
};
|
|
var expr = ol.expr.parse('foo(!bar)');
|
|
expect(expr).to.be.a(ol.expr.Call);
|
|
expect(expr.evaluate({bar: true}, lib)).to.be(false);
|
|
expect(expr.evaluate({bar: false}, lib)).to.be(true);
|
|
});
|
|
|
|
});
|
|
|
|
describe('11.5 - multiplicitave operators', function() {
|
|
// http://www.ecma-international.org/ecma-262/5.1/#sec-11.5
|
|
|
|
it('parses * operator', function() {
|
|
var expr = ol.expr.parse('foo*bar');
|
|
expect(expr).to.be.a(ol.expr.Math);
|
|
expect(expr.evaluate({foo: 10, bar: 20})).to.be(200);
|
|
});
|
|
|
|
it('consumes whitespace as expected with *', function() {
|
|
var expr = ol.expr.parse(' foo * bar ');
|
|
expect(expr).to.be.a(ol.expr.Math);
|
|
expect(expr.evaluate({foo: 15, bar: 2})).to.be(30);
|
|
});
|
|
|
|
it('parses / operator', function() {
|
|
var expr = ol.expr.parse('foo/12');
|
|
expect(expr).to.be.a(ol.expr.Math);
|
|
expect(expr.evaluate({foo: 10})).to.be(10 / 12);
|
|
});
|
|
|
|
it('consumes whitespace as expected with /', function() {
|
|
var expr = ol.expr.parse(' 4 / bar ');
|
|
expect(expr).to.be.a(ol.expr.Math);
|
|
expect(expr.evaluate({bar: 3})).to.be(4 / 3);
|
|
});
|
|
|
|
it('parses % operator', function() {
|
|
var expr = ol.expr.parse('12%foo');
|
|
expect(expr).to.be.a(ol.expr.Math);
|
|
expect(expr.evaluate({foo: 10})).to.be(2);
|
|
});
|
|
|
|
it('consumes whitespace as expected with %', function() {
|
|
var expr = ol.expr.parse(' 4 %bar ');
|
|
expect(expr).to.be.a(ol.expr.Math);
|
|
expect(expr.evaluate({bar: 3})).to.be(1);
|
|
});
|
|
|
|
it('parses * in call argument', function() {
|
|
var lib = {
|
|
foo: function(arg) {
|
|
return arg;
|
|
}
|
|
};
|
|
var expr = ol.expr.parse('foo(2 * bar)');
|
|
expect(expr).to.be.a(ol.expr.Call);
|
|
expect(expr.evaluate({bar: 3}, lib)).to.be(6);
|
|
expect(expr.evaluate({bar: 4}, lib)).to.be(8);
|
|
});
|
|
|
|
it('evaluates left to right for equal precedence', function() {
|
|
var expr = ol.expr.parse('2 / 4 * 20 % 15');
|
|
expect(expr.evaluate()).to.be(10);
|
|
});
|
|
|
|
it('respects group precedence', function() {
|
|
expect(ol.expr.parse('2 / 4 * (20 % 15)').evaluate()).to.be(2.5);
|
|
expect(ol.expr.parse('2 / (4 * (20 % 15))').evaluate()).to.be(0.1);
|
|
expect(ol.expr.parse('2 / ((4 * 20) % 15)').evaluate()).to.be(0.4);
|
|
expect(ol.expr.parse('2 / (4 * 20) % 15').evaluate()).to.be(0.025);
|
|
expect(ol.expr.parse('(2 / (4 * 20)) % 15').evaluate()).to.be(0.025);
|
|
expect(ol.expr.parse('(2 / 4) * 20 % 15').evaluate()).to.be(10);
|
|
expect(ol.expr.parse('((2 / 4) * 20) % 15').evaluate()).to.be(10);
|
|
});
|
|
|
|
it('parses * in left side of comparison expression', function() {
|
|
var expr = ol.expr.parse('foo * 2 >bar');
|
|
expect(expr).to.be.a(ol.expr.Comparison);
|
|
expect(expr.evaluate({foo: 4, bar: 7})).to.be(true);
|
|
expect(expr.evaluate({foo: 4, bar: 8})).to.be(false);
|
|
});
|
|
|
|
it('parses * in right side of comparison expression', function() {
|
|
var expr = ol.expr.parse('foo > 2 * bar');
|
|
expect(expr).to.be.a(ol.expr.Comparison);
|
|
expect(expr.evaluate({foo: 4, bar: 1})).to.be(true);
|
|
expect(expr.evaluate({foo: 4, bar: 2})).to.be(false);
|
|
});
|
|
|
|
});
|
|
|
|
describe('11.6 - additive operators', function() {
|
|
// http://www.ecma-international.org/ecma-262/5.1/#sec-11.6
|
|
|
|
it('parses + operator', function() {
|
|
var expr = ol.expr.parse('foo+bar');
|
|
expect(expr).to.be.a(ol.expr.Math);
|
|
expect(expr.evaluate({foo: 10, bar: 20})).to.be(30);
|
|
});
|
|
|
|
it('consumes whitespace as expected with +', function() {
|
|
var expr = ol.expr.parse(' foo +10 ');
|
|
expect(expr).to.be.a(ol.expr.Math);
|
|
expect(expr.evaluate({foo: 15})).to.be(25);
|
|
});
|
|
|
|
it('parses - operator', function() {
|
|
var expr = ol.expr.parse('foo-bar');
|
|
expect(expr).to.be.a(ol.expr.Math);
|
|
expect(expr.evaluate({foo: 10, bar: 20})).to.be(-10);
|
|
});
|
|
|
|
it('consumes whitespace as expected with -', function() {
|
|
var expr = ol.expr.parse(' foo- 10 ');
|
|
expect(expr).to.be.a(ol.expr.Math);
|
|
expect(expr.evaluate({foo: 15})).to.be(5);
|
|
});
|
|
|
|
it('respects precedence', function() {
|
|
expect(ol.expr.parse('2 + 4 * 20 - 15').evaluate()).to.be(67);
|
|
expect(ol.expr.parse('(2 + 4) * 20 - 15').evaluate()).to.be(105);
|
|
expect(ol.expr.parse('((2 + 4) * 20) - 15').evaluate()).to.be(105);
|
|
expect(ol.expr.parse('(2 + (4 * 20)) - 15').evaluate()).to.be(67);
|
|
expect(ol.expr.parse('2 + (4 * 20) - 15').evaluate()).to.be(67);
|
|
expect(ol.expr.parse('2 + ((4 * 20) - 15)').evaluate()).to.be(67);
|
|
expect(ol.expr.parse('2 + (4 * (20 - 15))').evaluate()).to.be(22);
|
|
expect(ol.expr.parse('2 + 4 * (20 - 15)').evaluate()).to.be(22);
|
|
});
|
|
|
|
it('parses + in call argument', function() {
|
|
var lib = {
|
|
foo: function(arg) {
|
|
return arg;
|
|
}
|
|
};
|
|
var expr = ol.expr.parse('foo(2 + bar)');
|
|
expect(expr).to.be.a(ol.expr.Call);
|
|
expect(expr.evaluate({bar: 3}, lib)).to.be(5);
|
|
expect(expr.evaluate({bar: 4}, lib)).to.be(6);
|
|
});
|
|
|
|
it('parses + in left side of comparison expression', function() {
|
|
var expr = ol.expr.parse('foo+2>bar');
|
|
expect(expr).to.be.a(ol.expr.Comparison);
|
|
expect(expr.evaluate({foo: 4, bar: 5})).to.be(true);
|
|
expect(expr.evaluate({foo: 4, bar: 6})).to.be(false);
|
|
});
|
|
|
|
it('parses + in right side of comparison expression', function() {
|
|
var expr = ol.expr.parse('foo >2 +bar');
|
|
expect(expr).to.be.a(ol.expr.Comparison);
|
|
expect(expr.evaluate({foo: 4, bar: 1})).to.be(true);
|
|
expect(expr.evaluate({foo: 4, bar: 2})).to.be(false);
|
|
});
|
|
|
|
});
|
|
|
|
describe('11.7 - bitwise shift operators', function() {
|
|
// http://www.ecma-international.org/ecma-262/5.1/#sec-11.7
|
|
it('not supported');
|
|
});
|
|
|
|
describe('11.8 - relational operators', function() {
|
|
// http://www.ecma-international.org/ecma-262/5.1/#sec-11.8
|
|
|
|
it('parses < operator', function() {
|
|
var expr = ol.expr.parse('foo<bar');
|
|
expect(expr).to.be.a(ol.expr.Comparison);
|
|
expect(expr.evaluate({foo: 10, bar: 20})).to.be(true);
|
|
expect(expr.evaluate({foo: 100, bar: 20})).to.be(false);
|
|
});
|
|
|
|
it('consumes whitespace as expected with <', function() {
|
|
var expr = ol.expr.parse(' foo <10 ');
|
|
expect(expr).to.be.a(ol.expr.Comparison);
|
|
expect(expr.evaluate({foo: 15})).to.be(false);
|
|
expect(expr.evaluate({foo: 5})).to.be(true);
|
|
});
|
|
|
|
it('parses > operator', function() {
|
|
var expr = ol.expr.parse('foo>bar');
|
|
expect(expr).to.be.a(ol.expr.Comparison);
|
|
expect(expr.evaluate({foo: 10, bar: 20})).to.be(false);
|
|
expect(expr.evaluate({foo: 100, bar: 20})).to.be(true);
|
|
});
|
|
|
|
it('consumes whitespace as expected with >', function() {
|
|
var expr = ol.expr.parse(' foo> 10 ');
|
|
expect(expr).to.be.a(ol.expr.Comparison);
|
|
expect(expr.evaluate({foo: 15})).to.be(true);
|
|
expect(expr.evaluate({foo: 5})).to.be(false);
|
|
});
|
|
|
|
it('parses <= operator', function() {
|
|
var expr = ol.expr.parse('foo<=bar');
|
|
expect(expr).to.be.a(ol.expr.Comparison);
|
|
expect(expr.evaluate({foo: 10, bar: 20})).to.be(true);
|
|
expect(expr.evaluate({foo: 100, bar: 20})).to.be(false);
|
|
expect(expr.evaluate({foo: 20, bar: 20})).to.be(true);
|
|
});
|
|
|
|
it('consumes whitespace as expected with <=', function() {
|
|
var expr = ol.expr.parse(' foo<= 10 ');
|
|
expect(expr).to.be.a(ol.expr.Comparison);
|
|
expect(expr.evaluate({foo: 15})).to.be(false);
|
|
expect(expr.evaluate({foo: 5})).to.be(true);
|
|
expect(expr.evaluate({foo: 10})).to.be(true);
|
|
});
|
|
|
|
it('throws for invalid spacing with <=', function() {
|
|
expect(function() {
|
|
ol.expr.parse(' foo< = 10 ');
|
|
}).throwException(function(err) {
|
|
expect(err).to.be.an(ol.expr.UnexpectedToken);
|
|
var token = err.token;
|
|
expect(token.value).to.be('=');
|
|
expect(token.index).to.be(6);
|
|
});
|
|
});
|
|
|
|
it('parses >= operator', function() {
|
|
var expr = ol.expr.parse('foo>=bar');
|
|
expect(expr).to.be.a(ol.expr.Comparison);
|
|
expect(expr.evaluate({foo: 10, bar: 20})).to.be(false);
|
|
expect(expr.evaluate({foo: 100, bar: 20})).to.be(true);
|
|
expect(expr.evaluate({foo: 20, bar: 20})).to.be(true);
|
|
});
|
|
|
|
it('consumes whitespace as expected with >=', function() {
|
|
var expr = ol.expr.parse(' foo >=10 ');
|
|
expect(expr).to.be.a(ol.expr.Comparison);
|
|
expect(expr.evaluate({foo: 15})).to.be(true);
|
|
expect(expr.evaluate({foo: 5})).to.be(false);
|
|
expect(expr.evaluate({foo: 10})).to.be(true);
|
|
});
|
|
|
|
it('throws for invalid spacing with >=', function() {
|
|
expect(function() {
|
|
ol.expr.parse(' 10 > =foo ');
|
|
}).throwException(function(err) {
|
|
expect(err).to.be.an(ol.expr.UnexpectedToken);
|
|
var token = err.token;
|
|
expect(token.value).to.be('=');
|
|
expect(token.index).to.be(6);
|
|
});
|
|
});
|
|
|
|
});
|
|
|
|
describe('11.9 - equality operators', function() {
|
|
// http://www.ecma-international.org/ecma-262/5.1/#sec-11.9
|
|
|
|
it('parses == operator', function() {
|
|
var expr = ol.expr.parse('foo==42');
|
|
expect(expr).to.be.a(ol.expr.Comparison);
|
|
expect(expr.evaluate({foo: 42})).to.be(true);
|
|
expect(expr.evaluate({foo: 41})).to.be(false);
|
|
expect(expr.evaluate({foo: '42'})).to.be(true);
|
|
});
|
|
|
|
it('consumes whitespace as expected with ==', function() {
|
|
var expr = ol.expr.parse(' 42 ==foo ');
|
|
expect(expr).to.be.a(ol.expr.Comparison);
|
|
expect(expr.evaluate({foo: 42})).to.be(true);
|
|
expect(expr.evaluate({foo: 41})).to.be(false);
|
|
expect(expr.evaluate({foo: '42'})).to.be(true);
|
|
});
|
|
|
|
it('throws for invalid spacing with ==', function() {
|
|
expect(function() {
|
|
ol.expr.parse(' 10 = =foo ');
|
|
}).throwException(function(err) {
|
|
expect(err).to.be.an(ol.expr.UnexpectedToken);
|
|
var token = err.token;
|
|
expect(token.value).to.be('=');
|
|
expect(token.index).to.be(4);
|
|
});
|
|
});
|
|
|
|
it('parses != operator', function() {
|
|
var expr = ol.expr.parse('foo!=42');
|
|
expect(expr).to.be.a(ol.expr.Comparison);
|
|
expect(expr.evaluate({foo: 42})).to.be(false);
|
|
expect(expr.evaluate({foo: 41})).to.be(true);
|
|
expect(expr.evaluate({foo: '42'})).to.be(false);
|
|
});
|
|
|
|
it('consumes whitespace as expected with !=', function() {
|
|
var expr = ol.expr.parse(' 42 !=foo ');
|
|
expect(expr).to.be.a(ol.expr.Comparison);
|
|
expect(expr.evaluate({foo: 42})).to.be(false);
|
|
expect(expr.evaluate({foo: 41})).to.be(true);
|
|
expect(expr.evaluate({foo: '42'})).to.be(false);
|
|
});
|
|
|
|
it('throws for invalid spacing with !=', function() {
|
|
expect(function() {
|
|
ol.expr.parse(' 10! =foo ');
|
|
}).throwException(function(err) {
|
|
expect(err).to.be.an(ol.expr.UnexpectedToken);
|
|
var token = err.token;
|
|
expect(token.value).to.be('!');
|
|
expect(token.index).to.be(3);
|
|
});
|
|
});
|
|
|
|
it('parses === operator', function() {
|
|
var expr = ol.expr.parse('42===foo');
|
|
expect(expr).to.be.a(ol.expr.Comparison);
|
|
expect(expr.evaluate({foo: 42})).to.be(true);
|
|
expect(expr.evaluate({foo: 41})).to.be(false);
|
|
expect(expr.evaluate({foo: '42'})).to.be(false);
|
|
});
|
|
|
|
it('consumes whitespace as expected with ===', function() {
|
|
var expr = ol.expr.parse(' foo ===42 ');
|
|
expect(expr).to.be.a(ol.expr.Comparison);
|
|
expect(expr.evaluate({foo: 42})).to.be(true);
|
|
expect(expr.evaluate({foo: 41})).to.be(false);
|
|
expect(expr.evaluate({foo: '42'})).to.be(false);
|
|
});
|
|
|
|
it('throws for invalid spacing with ===', function() {
|
|
expect(function() {
|
|
ol.expr.parse(' 10 = == foo ');
|
|
}).throwException(function(err) {
|
|
expect(err).to.be.an(ol.expr.UnexpectedToken);
|
|
var token = err.token;
|
|
expect(token.value).to.be('=');
|
|
expect(token.index).to.be(4);
|
|
});
|
|
});
|
|
|
|
it('parses !== operator', function() {
|
|
var expr = ol.expr.parse('foo!==42');
|
|
expect(expr).to.be.a(ol.expr.Comparison);
|
|
expect(expr.evaluate({foo: 42})).to.be(false);
|
|
expect(expr.evaluate({foo: 41})).to.be(true);
|
|
expect(expr.evaluate({foo: '42'})).to.be(true);
|
|
});
|
|
|
|
it('consumes whitespace as expected with !==', function() {
|
|
var expr = ol.expr.parse(' 42 !== foo ');
|
|
expect(expr).to.be.a(ol.expr.Comparison);
|
|
expect(expr.evaluate({foo: 42})).to.be(false);
|
|
expect(expr.evaluate({foo: 41})).to.be(true);
|
|
expect(expr.evaluate({foo: '42'})).to.be(true);
|
|
});
|
|
|
|
it('throws for invalid spacing with !==', function() {
|
|
expect(function() {
|
|
ol.expr.parse(' 10 != = foo ');
|
|
}).throwException(function(err) {
|
|
expect(err).to.be.an(ol.expr.UnexpectedToken);
|
|
var token = err.token;
|
|
expect(token.value).to.be('=');
|
|
expect(token.index).to.be(7);
|
|
});
|
|
});
|
|
});
|
|
|
|
describe('11.10 - binary bitwise operators', function() {
|
|
// http://www.ecma-international.org/ecma-262/5.1/#sec-11.10
|
|
it('not supported');
|
|
});
|
|
|
|
describe('11.11 - binary logical operators', function() {
|
|
// http://www.ecma-international.org/ecma-262/5.1/#sec-11.11
|
|
|
|
it('parses && operator', function() {
|
|
var expr = ol.expr.parse('foo&&bar');
|
|
expect(expr).to.be.a(ol.expr.Logical);
|
|
expect(expr.evaluate({foo: true, bar: true})).to.be(true);
|
|
expect(expr.evaluate({foo: true, bar: false})).to.be(false);
|
|
expect(expr.evaluate({foo: false, bar: true})).to.be(false);
|
|
expect(expr.evaluate({foo: false, bar: false})).to.be(false);
|
|
});
|
|
|
|
it('consumes space as expected with &&', function() {
|
|
var expr = ol.expr.parse(' foo && bar ');
|
|
expect(expr).to.be.a(ol.expr.Logical);
|
|
expect(expr.evaluate({foo: true, bar: true})).to.be(true);
|
|
expect(expr.evaluate({foo: true, bar: false})).to.be(false);
|
|
expect(expr.evaluate({foo: false, bar: true})).to.be(false);
|
|
expect(expr.evaluate({foo: false, bar: false})).to.be(false);
|
|
});
|
|
|
|
it('throws for invalid spacing with &&', function() {
|
|
expect(function() {
|
|
ol.expr.parse('true & & false');
|
|
}).throwException(function(err) {
|
|
expect(err).to.be.an(ol.expr.UnexpectedToken);
|
|
var token = err.token;
|
|
expect(token.value).to.be('&');
|
|
expect(token.index).to.be(5);
|
|
});
|
|
});
|
|
|
|
it('parses || operator', function() {
|
|
var expr = ol.expr.parse('foo||bar');
|
|
expect(expr).to.be.a(ol.expr.Logical);
|
|
expect(expr.evaluate({foo: true, bar: true})).to.be(true);
|
|
expect(expr.evaluate({foo: true, bar: false})).to.be(true);
|
|
expect(expr.evaluate({foo: false, bar: true})).to.be(true);
|
|
expect(expr.evaluate({foo: false, bar: false})).to.be(false);
|
|
});
|
|
|
|
it('consumes space as expected with ||', function() {
|
|
var expr = ol.expr.parse(' foo || bar ');
|
|
expect(expr).to.be.a(ol.expr.Logical);
|
|
expect(expr.evaluate({foo: true, bar: true})).to.be(true);
|
|
expect(expr.evaluate({foo: true, bar: false})).to.be(true);
|
|
expect(expr.evaluate({foo: false, bar: true})).to.be(true);
|
|
expect(expr.evaluate({foo: false, bar: false})).to.be(false);
|
|
});
|
|
|
|
it('throws for invalid spacing with ||', function() {
|
|
expect(function() {
|
|
ol.expr.parse('true | | false');
|
|
}).throwException(function(err) {
|
|
expect(err).to.be.an(ol.expr.UnexpectedToken);
|
|
var token = err.token;
|
|
expect(token.value).to.be('|');
|
|
expect(token.index).to.be(5);
|
|
});
|
|
});
|
|
|
|
});
|
|
|
|
describe('11.12 - conditional operator', function() {
|
|
// http://www.ecma-international.org/ecma-262/5.1/#sec-11.12
|
|
it('not supported');
|
|
});
|
|
|
|
describe('11.13 - assignment operators', function() {
|
|
// http://www.ecma-international.org/ecma-262/5.1/#sec-11.13
|
|
it('not supported');
|
|
});
|
|
|
|
describe('11.14 - comma operator', function() {
|
|
// http://www.ecma-international.org/ecma-262/5.1/#sec-11.14
|
|
it('not supported');
|
|
});
|
|
|
|
});
|
|
|
|
describe('ol.expr.lib', function() {
|
|
|
|
var parse = ol.expr.parse;
|
|
var evaluate = ol.expr.evaluateFeature;
|
|
|
|
describe('concat()', function() {
|
|
var feature = new ol.Feature({
|
|
str: 'bar',
|
|
num: 42,
|
|
bool: false,
|
|
nul: null
|
|
});
|
|
|
|
it('concatenates strings', function() {
|
|
expect(evaluate(parse('concat(str, "after")'), feature))
|
|
.to.be('barafter');
|
|
expect(evaluate(parse('concat("before", str)'), feature))
|
|
.to.be('beforebar');
|
|
expect(evaluate(parse('concat("a", str, "b")'), feature))
|
|
.to.be('abarb');
|
|
});
|
|
|
|
it('concatenates numbers as strings', function() {
|
|
expect(evaluate(parse('concat(num, 0)'), feature))
|
|
.to.be('420');
|
|
expect(evaluate(parse('concat(0, num)'), feature))
|
|
.to.be('042');
|
|
expect(evaluate(parse('concat(42, 42)'), feature))
|
|
.to.be('4242');
|
|
expect(evaluate(parse('concat(str, num)'), feature))
|
|
.to.be('bar42');
|
|
});
|
|
|
|
it('concatenates booleans as strings', function() {
|
|
expect(evaluate(parse('concat(bool, "foo")'), feature))
|
|
.to.be('falsefoo');
|
|
expect(evaluate(parse('concat(true, str)'), feature))
|
|
.to.be('truebar');
|
|
expect(evaluate(parse('concat(true, false)'), feature))
|
|
.to.be('truefalse');
|
|
});
|
|
|
|
it('concatenates nulls as strings', function() {
|
|
expect(evaluate(parse('concat(nul, "foo")'), feature))
|
|
.to.be('nullfoo');
|
|
expect(evaluate(parse('concat(str, null)'), feature))
|
|
.to.be('barnull');
|
|
});
|
|
|
|
});
|
|
|
|
describe('counter()', function() {
|
|
var counter = parse('counter()');
|
|
var counterWithStart = parse('counter(1000)');
|
|
var start = evaluate(counter);
|
|
expect(evaluate(counter).to.be(start + 1));
|
|
expect(evaluate(counter).to.be(start + 2));
|
|
expect(evaluate(counterWithStart).to.be(start + 3 + 1000));
|
|
});
|
|
|
|
describe('extent()', function() {
|
|
|
|
var nw = new ol.Feature({
|
|
geom: new ol.geom.Polygon([[
|
|
[-180, 90], [0, 90], [0, 0], [-180, 0], [-180, 90]
|
|
]])
|
|
});
|
|
|
|
var se = new ol.Feature({
|
|
geom: new ol.geom.Polygon([[
|
|
[180, -90], [0, -90], [0, 0], [180, 0], [180, -90]
|
|
]])
|
|
});
|
|
|
|
var north = parse('extent(-100, 40, 100, 60)');
|
|
var south = parse('extent(-100, -60, 100, -40)');
|
|
var east = parse('extent(80, -50, 100, 50)');
|
|
var west = parse('extent(-100, -50, -80, 50)');
|
|
|
|
it('evaluates to true for features within given extent', function() {
|
|
|
|
expect(evaluate(north, nw)).to.be(true);
|
|
expect(evaluate(south, nw)).to.be(false);
|
|
expect(evaluate(east, nw)).to.be(false);
|
|
expect(evaluate(west, nw)).to.be(true);
|
|
|
|
expect(evaluate(north, se)).to.be(false);
|
|
expect(evaluate(south, se)).to.be(true);
|
|
expect(evaluate(east, se)).to.be(true);
|
|
expect(evaluate(west, se)).to.be(false);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
describe('fid()', function() {
|
|
|
|
var one = new ol.Feature();
|
|
one.setId('one');
|
|
|
|
var two = new ol.Feature();
|
|
two.setId('two');
|
|
|
|
var three = new ol.Feature();
|
|
three.setId('three');
|
|
|
|
var four = new ol.Feature();
|
|
four.setId('four');
|
|
|
|
var odd = parse('fid("one", "three")');
|
|
var even = parse('fid("two", "four")');
|
|
var first = parse('fid("one")');
|
|
var last = parse('fid("four")');
|
|
var none = parse('fid("foo")');
|
|
|
|
it('evaluates to true if feature id matches', function() {
|
|
expect(evaluate(odd, one)).to.be(true);
|
|
expect(evaluate(odd, three)).to.be(true);
|
|
expect(evaluate(even, two)).to.be(true);
|
|
expect(evaluate(even, four)).to.be(true);
|
|
expect(evaluate(first, one)).to.be(true);
|
|
expect(evaluate(last, four)).to.be(true);
|
|
});
|
|
|
|
it('evaluates to false if feature id doesn\'t match', function() {
|
|
expect(evaluate(odd, two)).to.be(false);
|
|
expect(evaluate(odd, four)).to.be(false);
|
|
expect(evaluate(even, one)).to.be(false);
|
|
expect(evaluate(even, three)).to.be(false);
|
|
expect(evaluate(first, two)).to.be(false);
|
|
expect(evaluate(first, three)).to.be(false);
|
|
expect(evaluate(first, four)).to.be(false);
|
|
expect(evaluate(last, one)).to.be(false);
|
|
expect(evaluate(last, two)).to.be(false);
|
|
expect(evaluate(last, three)).to.be(false);
|
|
expect(evaluate(none, one)).to.be(false);
|
|
expect(evaluate(none, two)).to.be(false);
|
|
expect(evaluate(none, three)).to.be(false);
|
|
expect(evaluate(none, four)).to.be(false);
|
|
});
|
|
|
|
});
|
|
|
|
describe('geometryType()', function() {
|
|
|
|
var point = new ol.Feature({
|
|
geom: new ol.geom.Point([0, 0])
|
|
});
|
|
|
|
var line = new ol.Feature({
|
|
geom: new ol.geom.LineString([[180, -90], [-180, 90]])
|
|
});
|
|
|
|
var poly = new ol.Feature({
|
|
geom: new ol.geom.Polygon([[
|
|
[180, -90], [0, -90], [0, 0], [180, 0], [180, -90]
|
|
]])
|
|
});
|
|
|
|
var isPoint = parse('geometryType("point")');
|
|
var isLine = parse('geometryType("linestring")');
|
|
var isPoly = parse('geometryType("polygon")');
|
|
var pointOrPoly = parse('geometryType("point") || geometryType("polygon")');
|
|
|
|
it('distinguishes point features', function() {
|
|
expect(evaluate(isPoint, point)).to.be(true);
|
|
expect(evaluate(isPoint, line)).to.be(false);
|
|
expect(evaluate(isPoint, poly)).to.be(false);
|
|
});
|
|
|
|
it('distinguishes line features', function() {
|
|
expect(evaluate(isLine, point)).to.be(false);
|
|
expect(evaluate(isLine, line)).to.be(true);
|
|
expect(evaluate(isLine, poly)).to.be(false);
|
|
});
|
|
|
|
it('distinguishes polygon features', function() {
|
|
expect(evaluate(isPoly, point)).to.be(false);
|
|
expect(evaluate(isPoly, line)).to.be(false);
|
|
expect(evaluate(isPoly, poly)).to.be(true);
|
|
});
|
|
|
|
it('can be composed in a logical expression', function() {
|
|
expect(evaluate(pointOrPoly, point)).to.be(true);
|
|
expect(evaluate(pointOrPoly, line)).to.be(false);
|
|
expect(evaluate(pointOrPoly, poly)).to.be(true);
|
|
});
|
|
|
|
});
|
|
|
|
describe('like()', function() {
|
|
|
|
var one = new ol.Feature({foo: 'bar'});
|
|
var two = new ol.Feature({foo: 'baz'});
|
|
var three = new ol.Feature({foo: 'foo'});
|
|
|
|
var like = parse('like(foo, "ba")');
|
|
|
|
it('First and second feature match, third does not', function() {
|
|
expect(evaluate(like, one), true);
|
|
expect(evaluate(like, two), true);
|
|
expect(evaluate(like, three), false);
|
|
});
|
|
|
|
var uclike = parse('like(foo, "BA")');
|
|
it('Matchcase is true by default', function() {
|
|
expect(evaluate(uclike, one), false);
|
|
expect(evaluate(uclike, two), false);
|
|
expect(evaluate(uclike, three), false);
|
|
});
|
|
|
|
var ilike = parse('like(foo, "BA", "*", ".", "!", false)');
|
|
it('Using matchcase false, first two features match again', function() {
|
|
expect(evaluate(ilike, one), true);
|
|
expect(evaluate(ilike, two), true);
|
|
expect(evaluate(uclike, three), false);
|
|
});
|
|
|
|
});
|
|
|
|
describe('ieq()', function() {
|
|
|
|
var one = new ol.Feature({foo: 'Bar'});
|
|
var two = new ol.Feature({bar: 'Foo'});
|
|
|
|
var ieq1 = parse('ieq(foo, "bar")');
|
|
var ieq2 = parse('ieq("foo", bar)');
|
|
|
|
it('case-insensitive equality for an attribute', function() {
|
|
expect(evaluate(ieq1, one), true);
|
|
});
|
|
|
|
it('case-insensitive equality for an attribute as second argument',
|
|
function() {
|
|
expect(evaluate(ieq2, two), true);
|
|
});
|
|
|
|
});
|
|
|
|
describe('ineq()', function() {
|
|
|
|
var one = new ol.Feature({foo: 'Bar'});
|
|
var two = new ol.Feature({bar: 'Foo'});
|
|
|
|
var ieq1 = parse('ineq(foo, "bar")');
|
|
var ieq2 = parse('ineq("foo", bar)');
|
|
|
|
it('case-insensitive non-equality for an attribute', function() {
|
|
expect(evaluate(ieq1, one), false);
|
|
});
|
|
|
|
it('case-insensitive non-equality for an attribute as second argument',
|
|
function() {
|
|
expect(evaluate(ieq2, two), false);
|
|
});
|
|
|
|
});
|
|
|
|
describe('renderIntent()', function() {
|
|
|
|
var feature = new ol.Feature();
|
|
feature.renderIntent = 'foo';
|
|
|
|
var isFoo = parse('renderIntent("foo")');
|
|
var isBar = parse('renderIntent("bar")');
|
|
|
|
it('True when renderIntent matches', function() {
|
|
expect(evaluate(isFoo, feature), true);
|
|
});
|
|
|
|
it('False when renderIntent does not match', function() {
|
|
expect(evaluate(isBar, feature), false);
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
describe('ol.expr.register()', function() {
|
|
|
|
var spy;
|
|
beforeEach(function() {
|
|
spy = sinon.spy();
|
|
});
|
|
|
|
it('registers custom functions in ol.expr.lib', function() {
|
|
ol.expr.register('someFunc', spy);
|
|
expect(ol.expr.lib.someFunc).to.be(spy);
|
|
});
|
|
|
|
it('allows custom functions to be called', function() {
|
|
ol.expr.register('myFunc', spy);
|
|
var expr = ol.expr.parse('myFunc(42)');
|
|
expr.evaluate(null, ol.expr.lib);
|
|
expect(spy.calledOnce);
|
|
expect(spy.calledWithExactly(42));
|
|
});
|
|
|
|
it('allows custom functions to be called with identifiers', function() {
|
|
ol.expr.register('myFunc', spy);
|
|
var expr = ol.expr.parse('myFunc(foo, 42)');
|
|
expr.evaluate({foo: 'bar'}, ol.expr.lib);
|
|
expect(spy.calledOnce).to.be(true);
|
|
expect(spy.calledWithExactly('bar', 42)).to.be(true);
|
|
});
|
|
|
|
it('allows custom functions to be called with custom this obj', function() {
|
|
ol.expr.register('myFunc', spy);
|
|
var expr = ol.expr.parse('myFunc(foo, 42)');
|
|
var that = {};
|
|
expr.evaluate({foo: 'bar'}, ol.expr.lib, that);
|
|
expect(spy.calledOnce).to.be(true);
|
|
expect(spy.calledWithExactly('bar', 42)).to.be(true);
|
|
expect(spy.calledOn(that)).to.be(true);
|
|
});
|
|
|
|
it('allows overriding existing ol.expr.lib functions', function() {
|
|
var orig = ol.expr.lib.extent;
|
|
expect(orig).not.to.be(spy);
|
|
ol.expr.register('extent', spy);
|
|
expect(ol.expr.lib.extent).to.be(spy);
|
|
ol.expr.lib.extent = orig;
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
goog.require('ol.Feature');
|
|
goog.require('ol.expr');
|
|
goog.require('ol.expr.Call');
|
|
goog.require('ol.expr.Comparison');
|
|
goog.require('ol.expr.Expression');
|
|
goog.require('ol.expr.Identifier');
|
|
goog.require('ol.expr.Literal');
|
|
goog.require('ol.expr.Logical');
|
|
goog.require('ol.expr.Math');
|
|
goog.require('ol.expr.Member');
|
|
goog.require('ol.expr.Not');
|
|
goog.require('ol.expr.TokenType');
|
|
goog.require('ol.expr.UnexpectedToken');
|
|
goog.require('ol.geom.LineString');
|
|
goog.require('ol.geom.Point');
|
|
goog.require('ol.geom.Polygon');
|