1081 lines
32 KiB
JavaScript
1081 lines
32 KiB
JavaScript
(function(QUnit) {
|
|
|
|
var router = null;
|
|
var location = null;
|
|
var lastRoute = null;
|
|
var lastArgs = [];
|
|
|
|
var onRoute = function(routerParam, route, args) {
|
|
lastRoute = route;
|
|
lastArgs = args;
|
|
};
|
|
|
|
var Location = function(href) {
|
|
this.replace(href);
|
|
};
|
|
|
|
_.extend(Location.prototype, {
|
|
|
|
parser: document.createElement('a'),
|
|
|
|
replace: function(href) {
|
|
this.parser.href = href;
|
|
_.extend(this, _.pick(this.parser,
|
|
'href',
|
|
'hash',
|
|
'host',
|
|
'search',
|
|
'fragment',
|
|
'pathname',
|
|
'protocol'
|
|
));
|
|
|
|
// In IE, anchor.pathname does not contain a leading slash though
|
|
// window.location.pathname does.
|
|
if (!/^\//.test(this.pathname)) this.pathname = '/' + this.pathname;
|
|
},
|
|
|
|
toString: function() {
|
|
return this.href;
|
|
}
|
|
|
|
});
|
|
|
|
QUnit.module('Backbone.Router', {
|
|
|
|
beforeEach: function() {
|
|
location = new Location('http://example.com');
|
|
Backbone.history = _.extend(new Backbone.History, {location: location});
|
|
router = new Router({testing: 101});
|
|
Backbone.history.interval = 9;
|
|
Backbone.history.start({pushState: false});
|
|
lastRoute = null;
|
|
lastArgs = [];
|
|
Backbone.history.on('route', onRoute);
|
|
},
|
|
|
|
afterEach: function() {
|
|
Backbone.history.stop();
|
|
Backbone.history.off('route', onRoute);
|
|
}
|
|
|
|
});
|
|
|
|
var ExternalObject = {
|
|
value: 'unset',
|
|
|
|
routingFunction: function(value) {
|
|
this.value = value;
|
|
}
|
|
};
|
|
ExternalObject.routingFunction = _.bind(ExternalObject.routingFunction, ExternalObject);
|
|
|
|
var Router = Backbone.Router.extend({
|
|
|
|
count: 0,
|
|
|
|
routes: {
|
|
'noCallback': 'noCallback',
|
|
'counter': 'counter',
|
|
'search/:query': 'search',
|
|
'search/:query/p:page': 'search',
|
|
'charñ': 'charUTF',
|
|
'char%C3%B1': 'charEscaped',
|
|
'contacts': 'contacts',
|
|
'contacts/new': 'newContact',
|
|
'contacts/:id': 'loadContact',
|
|
'route-event/:arg': 'routeEvent',
|
|
'optional(/:item)': 'optionalItem',
|
|
'named/optional/(y:z)': 'namedOptional',
|
|
'splat/*args/end': 'splat',
|
|
':repo/compare/*from...*to': 'github',
|
|
'decode/:named/*splat': 'decode',
|
|
'*first/complex-*part/*rest': 'complex',
|
|
'query/:entity': 'query',
|
|
'function/:value': ExternalObject.routingFunction,
|
|
'*anything': 'anything'
|
|
},
|
|
|
|
preinitialize: function(options) {
|
|
this.testpreinit = 'foo';
|
|
},
|
|
|
|
initialize: function(options) {
|
|
this.testing = options.testing;
|
|
this.route('implicit', 'implicit');
|
|
},
|
|
|
|
counter: function() {
|
|
this.count++;
|
|
},
|
|
|
|
implicit: function() {
|
|
this.count++;
|
|
},
|
|
|
|
search: function(query, page) {
|
|
this.query = query;
|
|
this.page = page;
|
|
},
|
|
|
|
charUTF: function() {
|
|
this.charType = 'UTF';
|
|
},
|
|
|
|
charEscaped: function() {
|
|
this.charType = 'escaped';
|
|
},
|
|
|
|
contacts: function() {
|
|
this.contact = 'index';
|
|
},
|
|
|
|
newContact: function() {
|
|
this.contact = 'new';
|
|
},
|
|
|
|
loadContact: function() {
|
|
this.contact = 'load';
|
|
},
|
|
|
|
optionalItem: function(arg) {
|
|
this.arg = arg !== void 0 ? arg : null;
|
|
},
|
|
|
|
splat: function(args) {
|
|
this.args = args;
|
|
},
|
|
|
|
github: function(repo, from, to) {
|
|
this.repo = repo;
|
|
this.from = from;
|
|
this.to = to;
|
|
},
|
|
|
|
complex: function(first, part, rest) {
|
|
this.first = first;
|
|
this.part = part;
|
|
this.rest = rest;
|
|
},
|
|
|
|
query: function(entity, args) {
|
|
this.entity = entity;
|
|
this.queryArgs = args;
|
|
},
|
|
|
|
anything: function(whatever) {
|
|
this.anything = whatever;
|
|
},
|
|
|
|
namedOptional: function(z) {
|
|
this.z = z;
|
|
},
|
|
|
|
decode: function(named, path) {
|
|
this.named = named;
|
|
this.path = path;
|
|
},
|
|
|
|
routeEvent: function(arg) {
|
|
}
|
|
|
|
});
|
|
|
|
QUnit.test('initialize', function(assert) {
|
|
assert.expect(1);
|
|
assert.equal(router.testing, 101);
|
|
});
|
|
|
|
QUnit.test('preinitialize', function(assert) {
|
|
assert.expect(1);
|
|
assert.equal(router.testpreinit, 'foo');
|
|
});
|
|
|
|
QUnit.test('routes (simple)', function(assert) {
|
|
assert.expect(4);
|
|
location.replace('http://example.com#search/news');
|
|
Backbone.history.checkUrl();
|
|
assert.equal(router.query, 'news');
|
|
assert.equal(router.page, void 0);
|
|
assert.equal(lastRoute, 'search');
|
|
assert.equal(lastArgs[0], 'news');
|
|
});
|
|
|
|
QUnit.test('routes (simple, but unicode)', function(assert) {
|
|
assert.expect(4);
|
|
location.replace('http://example.com#search/тест');
|
|
Backbone.history.checkUrl();
|
|
assert.equal(router.query, 'тест');
|
|
assert.equal(router.page, void 0);
|
|
assert.equal(lastRoute, 'search');
|
|
assert.equal(lastArgs[0], 'тест');
|
|
});
|
|
|
|
QUnit.test('routes (two part)', function(assert) {
|
|
assert.expect(2);
|
|
location.replace('http://example.com#search/nyc/p10');
|
|
Backbone.history.checkUrl();
|
|
assert.equal(router.query, 'nyc');
|
|
assert.equal(router.page, '10');
|
|
});
|
|
|
|
QUnit.test('routes via navigate', function(assert) {
|
|
assert.expect(2);
|
|
Backbone.history.navigate('search/manhattan/p20', {trigger: true});
|
|
assert.equal(router.query, 'manhattan');
|
|
assert.equal(router.page, '20');
|
|
});
|
|
|
|
QUnit.test('routes via navigate with params', function(assert) {
|
|
assert.expect(1);
|
|
Backbone.history.navigate('query/test?a=b', {trigger: true});
|
|
assert.equal(router.queryArgs, 'a=b');
|
|
});
|
|
|
|
QUnit.test('routes via navigate for backwards-compatibility', function(assert) {
|
|
assert.expect(2);
|
|
Backbone.history.navigate('search/manhattan/p20', true);
|
|
assert.equal(router.query, 'manhattan');
|
|
assert.equal(router.page, '20');
|
|
});
|
|
|
|
QUnit.test('reports matched route via nagivate', function(assert) {
|
|
assert.expect(1);
|
|
assert.ok(Backbone.history.navigate('search/manhattan/p20', true));
|
|
});
|
|
|
|
QUnit.test('route precedence via navigate', function(assert) {
|
|
assert.expect(6);
|
|
|
|
// Check both 0.9.x and backwards-compatibility options
|
|
_.each([{trigger: true}, true], function(options) {
|
|
Backbone.history.navigate('contacts', options);
|
|
assert.equal(router.contact, 'index');
|
|
Backbone.history.navigate('contacts/new', options);
|
|
assert.equal(router.contact, 'new');
|
|
Backbone.history.navigate('contacts/foo', options);
|
|
assert.equal(router.contact, 'load');
|
|
});
|
|
});
|
|
|
|
QUnit.test('loadUrl is not called for identical routes.', function(assert) {
|
|
assert.expect(0);
|
|
Backbone.history.loadUrl = function() { assert.ok(false); };
|
|
location.replace('http://example.com#route');
|
|
Backbone.history.navigate('route');
|
|
Backbone.history.navigate('/route');
|
|
Backbone.history.navigate('/route');
|
|
});
|
|
|
|
QUnit.test('use implicit callback if none provided', function(assert) {
|
|
assert.expect(1);
|
|
router.count = 0;
|
|
router.navigate('implicit', {trigger: true});
|
|
assert.equal(router.count, 1);
|
|
});
|
|
|
|
QUnit.test('routes via navigate with {replace: true}', function(assert) {
|
|
assert.expect(1);
|
|
location.replace('http://example.com#start_here');
|
|
Backbone.history.checkUrl();
|
|
location.replace = function(href) {
|
|
assert.strictEqual(href, new Location('http://example.com#end_here').href);
|
|
};
|
|
Backbone.history.navigate('end_here', {replace: true});
|
|
});
|
|
|
|
QUnit.test('routes (splats)', function(assert) {
|
|
assert.expect(1);
|
|
location.replace('http://example.com#splat/long-list/of/splatted_99args/end');
|
|
Backbone.history.checkUrl();
|
|
assert.equal(router.args, 'long-list/of/splatted_99args');
|
|
});
|
|
|
|
QUnit.test('routes (github)', function(assert) {
|
|
assert.expect(3);
|
|
location.replace('http://example.com#backbone/compare/1.0...braddunbar:with/slash');
|
|
Backbone.history.checkUrl();
|
|
assert.equal(router.repo, 'backbone');
|
|
assert.equal(router.from, '1.0');
|
|
assert.equal(router.to, 'braddunbar:with/slash');
|
|
});
|
|
|
|
QUnit.test('routes (optional)', function(assert) {
|
|
assert.expect(2);
|
|
location.replace('http://example.com#optional');
|
|
Backbone.history.checkUrl();
|
|
assert.ok(!router.arg);
|
|
location.replace('http://example.com#optional/thing');
|
|
Backbone.history.checkUrl();
|
|
assert.equal(router.arg, 'thing');
|
|
});
|
|
|
|
QUnit.test('routes (complex)', function(assert) {
|
|
assert.expect(3);
|
|
location.replace('http://example.com#one/two/three/complex-part/four/five/six/seven');
|
|
Backbone.history.checkUrl();
|
|
assert.equal(router.first, 'one/two/three');
|
|
assert.equal(router.part, 'part');
|
|
assert.equal(router.rest, 'four/five/six/seven');
|
|
});
|
|
|
|
QUnit.test('routes (query)', function(assert) {
|
|
assert.expect(5);
|
|
location.replace('http://example.com#query/mandel?a=b&c=d');
|
|
Backbone.history.checkUrl();
|
|
assert.equal(router.entity, 'mandel');
|
|
assert.equal(router.queryArgs, 'a=b&c=d');
|
|
assert.equal(lastRoute, 'query');
|
|
assert.equal(lastArgs[0], 'mandel');
|
|
assert.equal(lastArgs[1], 'a=b&c=d');
|
|
});
|
|
|
|
QUnit.test('routes (anything)', function(assert) {
|
|
assert.expect(1);
|
|
location.replace('http://example.com#doesnt-match-a-route');
|
|
Backbone.history.checkUrl();
|
|
assert.equal(router.anything, 'doesnt-match-a-route');
|
|
});
|
|
|
|
QUnit.test('routes (function)', function(assert) {
|
|
assert.expect(3);
|
|
router.on('route', function(name) {
|
|
assert.ok(name === '');
|
|
});
|
|
assert.equal(ExternalObject.value, 'unset');
|
|
location.replace('http://example.com#function/set');
|
|
Backbone.history.checkUrl();
|
|
assert.equal(ExternalObject.value, 'set');
|
|
});
|
|
|
|
QUnit.test('Decode named parameters, not splats.', function(assert) {
|
|
assert.expect(2);
|
|
location.replace('http://example.com#decode/a%2Fb/c%2Fd/e');
|
|
Backbone.history.checkUrl();
|
|
assert.strictEqual(router.named, 'a/b');
|
|
assert.strictEqual(router.path, 'c/d/e');
|
|
});
|
|
|
|
QUnit.test('fires event when router doesn\'t have callback on it', function(assert) {
|
|
assert.expect(1);
|
|
router.on('route:noCallback', function() { assert.ok(true); });
|
|
location.replace('http://example.com#noCallback');
|
|
Backbone.history.checkUrl();
|
|
});
|
|
|
|
QUnit.test('No events are triggered if #execute returns false.', function(assert) {
|
|
assert.expect(1);
|
|
var MyRouter = Backbone.Router.extend({
|
|
|
|
routes: {
|
|
foo: function() {
|
|
assert.ok(true);
|
|
}
|
|
},
|
|
|
|
execute: function(callback, args) {
|
|
callback.apply(this, args);
|
|
return false;
|
|
}
|
|
|
|
});
|
|
|
|
var myRouter = new MyRouter;
|
|
|
|
myRouter.on('route route:foo', function() {
|
|
assert.ok(false);
|
|
});
|
|
|
|
Backbone.history.on('route', function() {
|
|
assert.ok(false);
|
|
});
|
|
|
|
location.replace('http://example.com#foo');
|
|
Backbone.history.checkUrl();
|
|
});
|
|
|
|
QUnit.test('#933, #908 - leading slash', function(assert) {
|
|
assert.expect(2);
|
|
location.replace('http://example.com/root/foo');
|
|
|
|
Backbone.history.stop();
|
|
Backbone.history = _.extend(new Backbone.History, {location: location});
|
|
Backbone.history.start({root: '/root', hashChange: false, silent: true});
|
|
assert.strictEqual(Backbone.history.getFragment(), 'foo');
|
|
|
|
Backbone.history.stop();
|
|
Backbone.history = _.extend(new Backbone.History, {location: location});
|
|
Backbone.history.start({root: '/root/', hashChange: false, silent: true});
|
|
assert.strictEqual(Backbone.history.getFragment(), 'foo');
|
|
});
|
|
|
|
QUnit.test('#967 - Route callback gets passed encoded values.', function(assert) {
|
|
assert.expect(3);
|
|
var route = 'has%2Fslash/complex-has%23hash/has%20space';
|
|
Backbone.history.navigate(route, {trigger: true});
|
|
assert.strictEqual(router.first, 'has/slash');
|
|
assert.strictEqual(router.part, 'has#hash');
|
|
assert.strictEqual(router.rest, 'has space');
|
|
});
|
|
|
|
QUnit.test('correctly handles URLs with % (#868)', function(assert) {
|
|
assert.expect(3);
|
|
location.replace('http://example.com#search/fat%3A1.5%25');
|
|
Backbone.history.checkUrl();
|
|
location.replace('http://example.com#search/fat');
|
|
Backbone.history.checkUrl();
|
|
assert.equal(router.query, 'fat');
|
|
assert.equal(router.page, void 0);
|
|
assert.equal(lastRoute, 'search');
|
|
});
|
|
|
|
QUnit.test('#2666 - Hashes with UTF8 in them.', function(assert) {
|
|
assert.expect(2);
|
|
Backbone.history.navigate('charñ', {trigger: true});
|
|
assert.equal(router.charType, 'UTF');
|
|
Backbone.history.navigate('char%C3%B1', {trigger: true});
|
|
assert.equal(router.charType, 'UTF');
|
|
});
|
|
|
|
QUnit.test('#1185 - Use pathname when hashChange is not wanted.', function(assert) {
|
|
assert.expect(1);
|
|
Backbone.history.stop();
|
|
location.replace('http://example.com/path/name#hash');
|
|
Backbone.history = _.extend(new Backbone.History, {location: location});
|
|
Backbone.history.start({hashChange: false});
|
|
var fragment = Backbone.history.getFragment();
|
|
assert.strictEqual(fragment, location.pathname.replace(/^\//, ''));
|
|
});
|
|
|
|
QUnit.test('#1206 - Strip leading slash before location.assign.', function(assert) {
|
|
assert.expect(1);
|
|
Backbone.history.stop();
|
|
location.replace('http://example.com/root/');
|
|
Backbone.history = _.extend(new Backbone.History, {location: location});
|
|
Backbone.history.start({hashChange: false, root: '/root/'});
|
|
location.assign = function(pathname) {
|
|
assert.strictEqual(pathname, '/root/fragment');
|
|
};
|
|
Backbone.history.navigate('/fragment');
|
|
});
|
|
|
|
QUnit.test('#1387 - Root fragment without trailing slash.', function(assert) {
|
|
assert.expect(1);
|
|
Backbone.history.stop();
|
|
location.replace('http://example.com/root');
|
|
Backbone.history = _.extend(new Backbone.History, {location: location});
|
|
Backbone.history.start({hashChange: false, root: '/root/', silent: true});
|
|
assert.strictEqual(Backbone.history.getFragment(), '');
|
|
});
|
|
|
|
QUnit.test('#1366 - History does not prepend root to fragment.', function(assert) {
|
|
assert.expect(2);
|
|
Backbone.history.stop();
|
|
location.replace('http://example.com/root/');
|
|
Backbone.history = _.extend(new Backbone.History, {
|
|
location: location,
|
|
history: {
|
|
pushState: function(state, title, url) {
|
|
assert.strictEqual(url, '/root/x');
|
|
}
|
|
}
|
|
});
|
|
Backbone.history.start({
|
|
root: '/root/',
|
|
pushState: true,
|
|
hashChange: false
|
|
});
|
|
Backbone.history.navigate('x');
|
|
assert.strictEqual(Backbone.history.fragment, 'x');
|
|
});
|
|
|
|
QUnit.test('Normalize root.', function(assert) {
|
|
assert.expect(1);
|
|
Backbone.history.stop();
|
|
location.replace('http://example.com/root');
|
|
Backbone.history = _.extend(new Backbone.History, {
|
|
location: location,
|
|
history: {
|
|
pushState: function(state, title, url) {
|
|
assert.strictEqual(url, '/root/fragment');
|
|
}
|
|
}
|
|
});
|
|
Backbone.history.start({
|
|
pushState: true,
|
|
root: '/root',
|
|
hashChange: false
|
|
});
|
|
Backbone.history.navigate('fragment');
|
|
});
|
|
|
|
QUnit.test('Normalize root.', function(assert) {
|
|
assert.expect(1);
|
|
Backbone.history.stop();
|
|
location.replace('http://example.com/root#fragment');
|
|
Backbone.history = _.extend(new Backbone.History, {
|
|
location: location,
|
|
history: {
|
|
pushState: function(state, title, url) {},
|
|
replaceState: function(state, title, url) {
|
|
assert.strictEqual(url, '/root/fragment');
|
|
}
|
|
}
|
|
});
|
|
Backbone.history.start({
|
|
pushState: true,
|
|
root: '/root'
|
|
});
|
|
});
|
|
|
|
QUnit.test('Normalize root.', function(assert) {
|
|
assert.expect(1);
|
|
Backbone.history.stop();
|
|
location.replace('http://example.com/root');
|
|
Backbone.history = _.extend(new Backbone.History, {location: location});
|
|
Backbone.history.loadUrl = function() { assert.ok(true); };
|
|
Backbone.history.start({
|
|
pushState: true,
|
|
root: '/root'
|
|
});
|
|
});
|
|
|
|
QUnit.test('Normalize root - leading slash.', function(assert) {
|
|
assert.expect(1);
|
|
Backbone.history.stop();
|
|
location.replace('http://example.com/root');
|
|
Backbone.history = _.extend(new Backbone.History, {
|
|
location: location,
|
|
history: {
|
|
pushState: function() {},
|
|
replaceState: function() {}
|
|
}
|
|
});
|
|
Backbone.history.start({root: 'root'});
|
|
assert.strictEqual(Backbone.history.root, '/root/');
|
|
});
|
|
|
|
QUnit.test('Transition from hashChange to pushState.', function(assert) {
|
|
assert.expect(1);
|
|
Backbone.history.stop();
|
|
location.replace('http://example.com/root#x/y');
|
|
Backbone.history = _.extend(new Backbone.History, {
|
|
location: location,
|
|
history: {
|
|
pushState: function() {},
|
|
replaceState: function(state, title, url) {
|
|
assert.strictEqual(url, '/root/x/y');
|
|
}
|
|
}
|
|
});
|
|
Backbone.history.start({
|
|
root: 'root',
|
|
pushState: true
|
|
});
|
|
});
|
|
|
|
QUnit.test('#1619: Router: Normalize empty root', function(assert) {
|
|
assert.expect(1);
|
|
Backbone.history.stop();
|
|
location.replace('http://example.com/');
|
|
Backbone.history = _.extend(new Backbone.History, {
|
|
location: location,
|
|
history: {
|
|
pushState: function() {},
|
|
replaceState: function() {}
|
|
}
|
|
});
|
|
Backbone.history.start({root: ''});
|
|
assert.strictEqual(Backbone.history.root, '/');
|
|
});
|
|
|
|
QUnit.test('#1619: Router: nagivate with empty root', function(assert) {
|
|
assert.expect(1);
|
|
Backbone.history.stop();
|
|
location.replace('http://example.com/');
|
|
Backbone.history = _.extend(new Backbone.History, {
|
|
location: location,
|
|
history: {
|
|
pushState: function(state, title, url) {
|
|
assert.strictEqual(url, '/fragment');
|
|
}
|
|
}
|
|
});
|
|
Backbone.history.start({
|
|
pushState: true,
|
|
root: '',
|
|
hashChange: false
|
|
});
|
|
Backbone.history.navigate('fragment');
|
|
});
|
|
|
|
QUnit.test('Transition from pushState to hashChange.', function(assert) {
|
|
assert.expect(1);
|
|
Backbone.history.stop();
|
|
location.replace('http://example.com/root/x/y?a=b');
|
|
location.replace = function(url) {
|
|
assert.strictEqual(url, '/root#x/y?a=b');
|
|
};
|
|
Backbone.history = _.extend(new Backbone.History, {
|
|
location: location,
|
|
history: {
|
|
pushState: null,
|
|
replaceState: null
|
|
}
|
|
});
|
|
Backbone.history.start({
|
|
root: 'root',
|
|
pushState: true
|
|
});
|
|
});
|
|
|
|
QUnit.test('#1695 - hashChange to pushState with search.', function(assert) {
|
|
assert.expect(1);
|
|
Backbone.history.stop();
|
|
location.replace('http://example.com/root#x/y?a=b');
|
|
Backbone.history = _.extend(new Backbone.History, {
|
|
location: location,
|
|
history: {
|
|
pushState: function() {},
|
|
replaceState: function(state, title, url) {
|
|
assert.strictEqual(url, '/root/x/y?a=b');
|
|
}
|
|
}
|
|
});
|
|
Backbone.history.start({
|
|
root: 'root',
|
|
pushState: true
|
|
});
|
|
});
|
|
|
|
QUnit.test('#1746 - Router allows empty route.', function(assert) {
|
|
assert.expect(1);
|
|
var MyRouter = Backbone.Router.extend({
|
|
routes: {'': 'empty'},
|
|
empty: function() {},
|
|
route: function(route) {
|
|
assert.strictEqual(route, '');
|
|
}
|
|
});
|
|
new MyRouter;
|
|
});
|
|
|
|
QUnit.test('#1794 - Trailing space in fragments.', function(assert) {
|
|
assert.expect(1);
|
|
var history = new Backbone.History;
|
|
assert.strictEqual(history.getFragment('fragment '), 'fragment');
|
|
});
|
|
|
|
QUnit.test('#1820 - Leading slash and trailing space.', function(assert) {
|
|
assert.expect(1);
|
|
var history = new Backbone.History;
|
|
assert.strictEqual(history.getFragment('/fragment '), 'fragment');
|
|
});
|
|
|
|
QUnit.test('#1980 - Optional parameters.', function(assert) {
|
|
assert.expect(2);
|
|
location.replace('http://example.com#named/optional/y');
|
|
Backbone.history.checkUrl();
|
|
assert.strictEqual(router.z, undefined);
|
|
location.replace('http://example.com#named/optional/y123');
|
|
Backbone.history.checkUrl();
|
|
assert.strictEqual(router.z, '123');
|
|
});
|
|
|
|
QUnit.test('#2062 - Trigger "route" event on router instance.', function(assert) {
|
|
assert.expect(2);
|
|
router.on('route', function(name, args) {
|
|
assert.strictEqual(name, 'routeEvent');
|
|
assert.deepEqual(args, ['x', null]);
|
|
});
|
|
location.replace('http://example.com#route-event/x');
|
|
Backbone.history.checkUrl();
|
|
});
|
|
|
|
QUnit.test('#2255 - Extend routes by making routes a function.', function(assert) {
|
|
assert.expect(1);
|
|
var RouterBase = Backbone.Router.extend({
|
|
routes: function() {
|
|
return {
|
|
home: 'root',
|
|
index: 'index.html'
|
|
};
|
|
}
|
|
});
|
|
|
|
var RouterExtended = RouterBase.extend({
|
|
routes: function() {
|
|
var _super = RouterExtended.__super__.routes;
|
|
return _.extend(_super(), {show: 'show', search: 'search'});
|
|
}
|
|
});
|
|
|
|
var myRouter = new RouterExtended();
|
|
assert.deepEqual({home: 'root', index: 'index.html', show: 'show', search: 'search'}, myRouter.routes);
|
|
});
|
|
|
|
QUnit.test('#2538 - hashChange to pushState only if both requested.', function(assert) {
|
|
assert.expect(0);
|
|
Backbone.history.stop();
|
|
location.replace('http://example.com/root?a=b#x/y');
|
|
Backbone.history = _.extend(new Backbone.History, {
|
|
location: location,
|
|
history: {
|
|
pushState: function() {},
|
|
replaceState: function() { assert.ok(false); }
|
|
}
|
|
});
|
|
Backbone.history.start({
|
|
root: 'root',
|
|
pushState: true,
|
|
hashChange: false
|
|
});
|
|
});
|
|
|
|
QUnit.test('No hash fallback.', function(assert) {
|
|
assert.expect(0);
|
|
Backbone.history.stop();
|
|
Backbone.history = _.extend(new Backbone.History, {
|
|
location: location,
|
|
history: {
|
|
pushState: function() {},
|
|
replaceState: function() {}
|
|
}
|
|
});
|
|
|
|
var MyRouter = Backbone.Router.extend({
|
|
routes: {
|
|
hash: function() { assert.ok(false); }
|
|
}
|
|
});
|
|
var myRouter = new MyRouter;
|
|
|
|
location.replace('http://example.com/');
|
|
Backbone.history.start({
|
|
pushState: true,
|
|
hashChange: false
|
|
});
|
|
location.replace('http://example.com/nomatch#hash');
|
|
Backbone.history.checkUrl();
|
|
});
|
|
|
|
QUnit.test('#2656 - No trailing slash on root.', function(assert) {
|
|
assert.expect(1);
|
|
Backbone.history.stop();
|
|
Backbone.history = _.extend(new Backbone.History, {
|
|
location: location,
|
|
history: {
|
|
pushState: function(state, title, url) {
|
|
assert.strictEqual(url, '/root');
|
|
}
|
|
}
|
|
});
|
|
location.replace('http://example.com/root/path');
|
|
Backbone.history.start({pushState: true, hashChange: false, root: 'root'});
|
|
Backbone.history.navigate('');
|
|
});
|
|
|
|
QUnit.test('#2656 - No trailing slash on root.', function(assert) {
|
|
assert.expect(1);
|
|
Backbone.history.stop();
|
|
Backbone.history = _.extend(new Backbone.History, {
|
|
location: location,
|
|
history: {
|
|
pushState: function(state, title, url) {
|
|
assert.strictEqual(url, '/');
|
|
}
|
|
}
|
|
});
|
|
location.replace('http://example.com/path');
|
|
Backbone.history.start({pushState: true, hashChange: false});
|
|
Backbone.history.navigate('');
|
|
});
|
|
|
|
QUnit.test('#2656 - No trailing slash on root.', function(assert) {
|
|
assert.expect(1);
|
|
Backbone.history.stop();
|
|
Backbone.history = _.extend(new Backbone.History, {
|
|
location: location,
|
|
history: {
|
|
pushState: function(state, title, url) {
|
|
assert.strictEqual(url, '/root?x=1');
|
|
}
|
|
}
|
|
});
|
|
location.replace('http://example.com/root/path');
|
|
Backbone.history.start({pushState: true, hashChange: false, root: 'root'});
|
|
Backbone.history.navigate('?x=1');
|
|
});
|
|
|
|
QUnit.test('#2765 - Fragment matching sans query/hash.', function(assert) {
|
|
assert.expect(2);
|
|
Backbone.history.stop();
|
|
Backbone.history = _.extend(new Backbone.History, {
|
|
location: location,
|
|
history: {
|
|
pushState: function(state, title, url) {
|
|
assert.strictEqual(url, '/path?query#hash');
|
|
}
|
|
}
|
|
});
|
|
|
|
var MyRouter = Backbone.Router.extend({
|
|
routes: {
|
|
path: function() { assert.ok(true); }
|
|
}
|
|
});
|
|
var myRouter = new MyRouter;
|
|
|
|
location.replace('http://example.com/');
|
|
Backbone.history.start({pushState: true, hashChange: false});
|
|
Backbone.history.navigate('path?query#hash', true);
|
|
});
|
|
|
|
QUnit.test('Do not decode the search params.', function(assert) {
|
|
assert.expect(1);
|
|
var MyRouter = Backbone.Router.extend({
|
|
routes: {
|
|
path: function(params) {
|
|
assert.strictEqual(params, 'x=y%3Fz');
|
|
}
|
|
}
|
|
});
|
|
var myRouter = new MyRouter;
|
|
Backbone.history.navigate('path?x=y%3Fz', true);
|
|
});
|
|
|
|
QUnit.test('Navigate to a hash url.', function(assert) {
|
|
assert.expect(1);
|
|
Backbone.history.stop();
|
|
Backbone.history = _.extend(new Backbone.History, {location: location});
|
|
Backbone.history.start({pushState: true});
|
|
var MyRouter = Backbone.Router.extend({
|
|
routes: {
|
|
path: function(params) {
|
|
assert.strictEqual(params, 'x=y');
|
|
}
|
|
}
|
|
});
|
|
var myRouter = new MyRouter;
|
|
location.replace('http://example.com/path?x=y#hash');
|
|
Backbone.history.checkUrl();
|
|
});
|
|
|
|
QUnit.test('#navigate to a hash url.', function(assert) {
|
|
assert.expect(1);
|
|
Backbone.history.stop();
|
|
Backbone.history = _.extend(new Backbone.History, {location: location});
|
|
Backbone.history.start({pushState: true});
|
|
var MyRouter = Backbone.Router.extend({
|
|
routes: {
|
|
path: function(params) {
|
|
assert.strictEqual(params, 'x=y');
|
|
}
|
|
}
|
|
});
|
|
var myRouter = new MyRouter;
|
|
Backbone.history.navigate('path?x=y#hash', true);
|
|
});
|
|
|
|
QUnit.test('unicode pathname', function(assert) {
|
|
assert.expect(1);
|
|
location.replace('http://example.com/myyjä');
|
|
Backbone.history.stop();
|
|
Backbone.history = _.extend(new Backbone.History, {location: location});
|
|
var MyRouter = Backbone.Router.extend({
|
|
routes: {
|
|
myyjä: function() {
|
|
assert.ok(true);
|
|
}
|
|
}
|
|
});
|
|
new MyRouter;
|
|
Backbone.history.start({pushState: true});
|
|
});
|
|
|
|
QUnit.test('unicode pathname with % in a parameter', function(assert) {
|
|
assert.expect(1);
|
|
location.replace('http://example.com/myyjä/foo%20%25%3F%2f%40%25%20bar');
|
|
location.pathname = '/myyj%C3%A4/foo%20%25%3F%2f%40%25%20bar';
|
|
Backbone.history.stop();
|
|
Backbone.history = _.extend(new Backbone.History, {location: location});
|
|
var MyRouter = Backbone.Router.extend({
|
|
routes: {
|
|
'myyjä/:query': function(query) {
|
|
assert.strictEqual(query, 'foo %?/@% bar');
|
|
}
|
|
}
|
|
});
|
|
new MyRouter;
|
|
Backbone.history.start({pushState: true});
|
|
});
|
|
|
|
QUnit.test('newline in route', function(assert) {
|
|
assert.expect(1);
|
|
location.replace('http://example.com/stuff%0Anonsense?param=foo%0Abar');
|
|
Backbone.history.stop();
|
|
Backbone.history = _.extend(new Backbone.History, {location: location});
|
|
var MyRouter = Backbone.Router.extend({
|
|
routes: {
|
|
'stuff\nnonsense': function() {
|
|
assert.ok(true);
|
|
}
|
|
}
|
|
});
|
|
new MyRouter;
|
|
Backbone.history.start({pushState: true});
|
|
});
|
|
|
|
QUnit.test('Router#execute receives callback, args, name.', function(assert) {
|
|
assert.expect(3);
|
|
location.replace('http://example.com#foo/123/bar?x=y');
|
|
Backbone.history.stop();
|
|
Backbone.history = _.extend(new Backbone.History, {location: location});
|
|
var MyRouter = Backbone.Router.extend({
|
|
routes: {'foo/:id/bar': 'foo'},
|
|
foo: function() {},
|
|
execute: function(callback, args, name) {
|
|
assert.strictEqual(callback, this.foo);
|
|
assert.deepEqual(args, ['123', 'x=y']);
|
|
assert.strictEqual(name, 'foo');
|
|
}
|
|
});
|
|
var myRouter = new MyRouter;
|
|
Backbone.history.start();
|
|
});
|
|
|
|
QUnit.test('pushState to hashChange with only search params.', function(assert) {
|
|
assert.expect(1);
|
|
Backbone.history.stop();
|
|
location.replace('http://example.com?a=b');
|
|
location.replace = function(url) {
|
|
assert.strictEqual(url, '/#?a=b');
|
|
};
|
|
Backbone.history = _.extend(new Backbone.History, {
|
|
location: location,
|
|
history: null
|
|
});
|
|
Backbone.history.start({pushState: true});
|
|
});
|
|
|
|
QUnit.test('#3123 - History#navigate decodes before comparison.', function(assert) {
|
|
assert.expect(1);
|
|
Backbone.history.stop();
|
|
location.replace('http://example.com/shop/search?keyword=short%20dress');
|
|
Backbone.history = _.extend(new Backbone.History, {
|
|
location: location,
|
|
history: {
|
|
pushState: function() { assert.ok(false); },
|
|
replaceState: function() { assert.ok(false); }
|
|
}
|
|
});
|
|
Backbone.history.start({pushState: true});
|
|
Backbone.history.navigate('shop/search?keyword=short%20dress', true);
|
|
assert.strictEqual(Backbone.history.fragment, 'shop/search?keyword=short dress');
|
|
});
|
|
|
|
QUnit.test('#3175 - Urls in the params', function(assert) {
|
|
assert.expect(1);
|
|
Backbone.history.stop();
|
|
location.replace('http://example.com#login?a=value&backUrl=https%3A%2F%2Fwww.msn.com%2Fidp%2Fidpdemo%3Fspid%3Dspdemo%26target%3Db');
|
|
Backbone.history = _.extend(new Backbone.History, {location: location});
|
|
var myRouter = new Backbone.Router;
|
|
myRouter.route('login', function(params) {
|
|
assert.strictEqual(params, 'a=value&backUrl=https%3A%2F%2Fwww.msn.com%2Fidp%2Fidpdemo%3Fspid%3Dspdemo%26target%3Db');
|
|
});
|
|
Backbone.history.start();
|
|
});
|
|
|
|
QUnit.test('#3358 - pushState to hashChange transition with search params', function(assert) {
|
|
assert.expect(1);
|
|
Backbone.history.stop();
|
|
location.replace('http://example.com/root?foo=bar');
|
|
location.replace = function(url) {
|
|
assert.strictEqual(url, '/root#?foo=bar');
|
|
};
|
|
Backbone.history = _.extend(new Backbone.History, {
|
|
location: location,
|
|
history: {
|
|
pushState: undefined,
|
|
replaceState: undefined
|
|
}
|
|
});
|
|
Backbone.history.start({root: '/root', pushState: true});
|
|
});
|
|
|
|
QUnit.test('Paths that don\'t match the root should not match no root', function(assert) {
|
|
assert.expect(0);
|
|
location.replace('http://example.com/foo');
|
|
Backbone.history.stop();
|
|
Backbone.history = _.extend(new Backbone.History, {location: location});
|
|
var MyRouter = Backbone.Router.extend({
|
|
routes: {
|
|
foo: function() {
|
|
assert.ok(false, 'should not match unless root matches');
|
|
}
|
|
}
|
|
});
|
|
var myRouter = new MyRouter;
|
|
Backbone.history.start({root: 'root', pushState: true});
|
|
});
|
|
|
|
QUnit.test('Paths that don\'t match the root should not match roots of the same length', function(assert) {
|
|
assert.expect(0);
|
|
location.replace('http://example.com/xxxx/foo');
|
|
Backbone.history.stop();
|
|
Backbone.history = _.extend(new Backbone.History, {location: location});
|
|
var MyRouter = Backbone.Router.extend({
|
|
routes: {
|
|
foo: function() {
|
|
assert.ok(false, 'should not match unless root matches');
|
|
}
|
|
}
|
|
});
|
|
var myRouter = new MyRouter;
|
|
Backbone.history.start({root: 'root', pushState: true});
|
|
});
|
|
|
|
QUnit.test('roots with regex characters', function(assert) {
|
|
assert.expect(1);
|
|
location.replace('http://example.com/x+y.z/foo');
|
|
Backbone.history.stop();
|
|
Backbone.history = _.extend(new Backbone.History, {location: location});
|
|
var MyRouter = Backbone.Router.extend({
|
|
routes: {foo: function() { assert.ok(true); }}
|
|
});
|
|
var myRouter = new MyRouter;
|
|
Backbone.history.start({root: 'x+y.z', pushState: true});
|
|
});
|
|
|
|
QUnit.test('roots with unicode characters', function(assert) {
|
|
assert.expect(1);
|
|
location.replace('http://example.com/®ooτ/foo');
|
|
Backbone.history.stop();
|
|
Backbone.history = _.extend(new Backbone.History, {location: location});
|
|
var MyRouter = Backbone.Router.extend({
|
|
routes: {foo: function() { assert.ok(true); }}
|
|
});
|
|
var myRouter = new MyRouter;
|
|
Backbone.history.start({root: '®ooτ', pushState: true});
|
|
});
|
|
|
|
QUnit.test('roots without slash', function(assert) {
|
|
assert.expect(1);
|
|
location.replace('http://example.com/®ooτ');
|
|
Backbone.history.stop();
|
|
Backbone.history = _.extend(new Backbone.History, {location: location});
|
|
var MyRouter = Backbone.Router.extend({
|
|
routes: {'': function() { assert.ok(true); }}
|
|
});
|
|
var myRouter = new MyRouter;
|
|
Backbone.history.start({root: '®ooτ', pushState: true});
|
|
});
|
|
|
|
QUnit.test('#4025 - navigate updates URL hash as is', function(assert) {
|
|
assert.expect(1);
|
|
var route = 'search/has%20space';
|
|
Backbone.history.navigate(route);
|
|
assert.strictEqual(location.hash, '#' + route);
|
|
});
|
|
|
|
})(QUnit);
|