mirror of https://github.com/perkeep/perkeep.git
147 lines
4.6 KiB
JavaScript
147 lines
4.6 KiB
JavaScript
/*
|
|
Copyright 2014 The Perkeep Authors
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
you may not use this file except in compliance with the License.
|
|
You may obtain a copy of the License at
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
See the License for the specific language governing permissions and
|
|
limitations under the License.
|
|
*/
|
|
|
|
goog.require('goog.Uri');
|
|
goog.require('goog.events.EventTarget');
|
|
var assert = require('assert');
|
|
|
|
goog.require('cam.Navigator');
|
|
|
|
var MockLocation = function() {
|
|
goog.base(this);
|
|
this.href = '';
|
|
this.reloadCount = 0;
|
|
};
|
|
goog.inherits(MockLocation, goog.events.EventTarget);
|
|
MockLocation.prototype.reload = function() {
|
|
this.reloadCount++;
|
|
};
|
|
|
|
var MockHistory = function() {
|
|
this.states = [null];
|
|
};
|
|
MockHistory.prototype.pushState = function(a, b, url) {
|
|
this.states.push({state:a, url:url});
|
|
};
|
|
MockHistory.prototype.replaceState = function(a, b, url) {
|
|
this.states[this.states.length - 1] = {state:a, url:url};
|
|
}
|
|
|
|
var Handler = function() {
|
|
this.lastURL = null;
|
|
this.returnsTrue = false;
|
|
this.handle = this.handle.bind(this);
|
|
};
|
|
Handler.prototype.handle = function(url) {
|
|
this.lastURL = url;
|
|
return this.returnsTrue;
|
|
};
|
|
|
|
describe('cam.Navigator', function() {
|
|
var mockWindow, mockLocation, mockHistory, handler, navigator;
|
|
var url = new goog.Uri('http://www.camlistore.org/foobar');
|
|
|
|
beforeEach(function() {
|
|
mockWindow = new goog.events.EventTarget();
|
|
mockLocation = new MockLocation();
|
|
mockHistory = new MockHistory();
|
|
handler = new Handler();
|
|
navigator = new cam.Navigator(mockWindow, mockLocation, mockHistory);
|
|
navigator.onWillNavigate = handler.handle;
|
|
});
|
|
|
|
it ('#constructor - seed initial state', function() {
|
|
assert.deepEqual(mockHistory.states, [{state:{}, url:''}]);
|
|
});
|
|
|
|
it('#navigate - no handler', function() {
|
|
// We should do network navigation.
|
|
navigator.onWillNavigate = function(){};
|
|
var handledLocally = navigator.navigate(url);
|
|
assert.equal(mockLocation.href, url.toString());
|
|
assert.equal(mockHistory.states.length, 1);
|
|
assert.equal(handledLocally, false);
|
|
});
|
|
|
|
it('#navigate - handler returns false', function() {
|
|
// Both handlers should get called, we should do network navigation.
|
|
var handledLocally = navigator.navigate(url);
|
|
assert.equal(handler.lastURL, url);
|
|
assert.equal(mockLocation.href, url.toString());
|
|
assert.equal(mockHistory.states.length, 1);
|
|
assert.equal(handledLocally, false);
|
|
});
|
|
|
|
it('#navigate - handler returns true', function() {
|
|
// Both handlers should get called, we should do pushState() navigation.
|
|
handler.returnsTrue = true;
|
|
var handledLocally = navigator.navigate(url);
|
|
assert.equal(handler.lastURL, url);
|
|
assert.equal(mockLocation.href, '');
|
|
assert.deepEqual(mockHistory.states, [{state:{}, url:''}, {state:{}, url:url.toString()}]);
|
|
assert.equal(handledLocally, true);
|
|
});
|
|
|
|
it('#handleClick_ - handled', function() {
|
|
handler.returnsTrue = true;
|
|
var ev = new goog.events.Event('click');
|
|
ev.button = 0;
|
|
ev.target = {
|
|
nodeName: 'A',
|
|
href: url.toString()
|
|
};
|
|
mockWindow.dispatchEvent(ev);
|
|
assert.equal(mockLocation.href, '');
|
|
assert.deepEqual(mockHistory.states, [{state:{}, url:''}, {state:{}, url:url.toString()}]);
|
|
});
|
|
|
|
it('#handleClick_ - not handled', function() {
|
|
var ev = new goog.events.Event('click');
|
|
ev.button = 0;
|
|
ev.target = {
|
|
nodeName: 'A',
|
|
href: url.toString()
|
|
};
|
|
mockWindow.dispatchEvent(ev);
|
|
assert.equal(mockLocation.href, '');
|
|
assert.deepEqual(mockHistory.states, [{state:{}, url:''}]);
|
|
assert.equal(ev.defaultPrevented, false);
|
|
});
|
|
|
|
it('#handlePopState_ - handled', function() {
|
|
handler.returnsTrue = true;
|
|
mockWindow.dispatchEvent({type:'popstate', state:{}});
|
|
assert.equal(mockLocation.reloadCount, 0);
|
|
assert.deepEqual(mockHistory.states, [{state:{}, url:''}]);
|
|
});
|
|
|
|
it('#handlePopState_ - not handled', function() {
|
|
mockWindow.dispatchEvent({type:'popstate', state:{}});
|
|
assert.equal(mockLocation.reloadCount, 1);
|
|
assert.deepEqual(mockHistory.states, [{state:{}, url:''}]);
|
|
});
|
|
|
|
it('#handlePopState_ - ignore initial popstate', function() {
|
|
// Fire a popstate with no state property. This simulates what happens in buggy browsers onload. This one should be ignored.
|
|
mockWindow.dispatchEvent({type:'popstate', state:null});
|
|
assert.equal(mockLocation.reloadCount, 0);
|
|
|
|
// Now fire a popstate with a state property it should be handled.
|
|
mockWindow.dispatchEvent({type:'popstate', state:{}});
|
|
assert.equal(mockLocation.reloadCount, 1);
|
|
});
|
|
});
|