blob: 1de3c25e33f154494c18bb8eb837527065811046 [file] [log] [blame]
describe('tabs', function() {
var elm, scope;
beforeEach(module('ui.bootstrap.tabs'));
beforeEach(module('template/tabs/tabset.html'));
beforeEach(module('template/tabs/tab.html'));
function titles() {
return elm.find('ul.nav-tabs li');
}
function contents() {
return elm.find('div.tab-content div.tab-pane');
}
function expectTitles(titlesArray) {
var t = titles();
expect(t.length).toEqual(titlesArray.length);
for (var i=0; i<t.length; i++) {
expect(t.eq(i).text().trim()).toEqual(titlesArray[i]);
}
}
function expectContents(contentsArray) {
var c = contents();
expect(c.length).toEqual(contentsArray.length);
for (var i=0; i<c.length; i++) {
expect(c.eq(i).text().trim()).toEqual(contentsArray[i]);
}
}
describe('basics', function() {
beforeEach(inject(function($compile, $rootScope) {
scope = $rootScope.$new();
scope.first = '1';
scope.second = '2';
scope.actives = {};
scope.selectFirst = jasmine.createSpy();
scope.selectSecond = jasmine.createSpy();
scope.deselectFirst = jasmine.createSpy();
scope.deselectSecond = jasmine.createSpy();
elm = $compile([
'<uib-tabset class="hello" data-pizza="pepperoni">',
' <uib-tab heading="First Tab {{first}}" active="actives.one" select="selectFirst()" deselect="deselectFirst()">',
' first content is {{first}}',
' </uib-tab>',
' <uib-tab active="actives.two" select="selectSecond()" deselect="deselectSecond()">',
' <uib-tab-heading><b>Second</b> Tab {{second}}</uib-tab-heading>',
' second content is {{second}}',
' </uib-tab>',
'</uib-tabset>'
].join('\n'))(scope);
scope.$apply();
return elm;
}));
it('should pass class and other attributes on to tab template', function() {
expect(elm).toHaveClass('hello');
expect(elm.attr('data-pizza')).toBe('pepperoni');
});
it('should create clickable titles', function() {
var t = titles();
expect(t.length).toBe(2);
expect(t.find('a').eq(0).text()).toBe('First Tab 1');
//It should put the uib-tab-heading element into the 'a' title
expect(t.find('a').eq(1).children().is('uib-tab-heading')).toBe(true);
expect(t.find('a').eq(1).children().html()).toBe('<b>Second</b> Tab 2');
});
it('should bind tabs content and set first tab active', function() {
expectContents(['first content is 1', 'second content is 2']);
expect(titles().eq(0)).toHaveClass('active');
expect(titles().eq(1)).not.toHaveClass('active');
expect(scope.actives.one).toBe(true);
expect(scope.actives.two).toBeFalsy();
});
it('should change active on click', function() {
titles().eq(1).find('a').click();
expect(contents().eq(1)).toHaveClass('active');
expect(titles().eq(0)).not.toHaveClass('active');
expect(titles().eq(1)).toHaveClass('active');
expect(scope.actives.one).toBe(false);
expect(scope.actives.two).toBe(true);
});
it('should call select callback on select', function() {
titles().eq(1).find('a').click();
expect(scope.selectSecond).toHaveBeenCalled();
titles().eq(0).find('a').click();
expect(scope.selectFirst).toHaveBeenCalled();
});
it('should call deselect callback on deselect', function() {
titles().eq(1).find('a').click();
titles().eq(0).find('a').click();
expect(scope.deselectSecond).toHaveBeenCalled();
titles().eq(1).find('a').click();
expect(scope.deselectFirst).toHaveBeenCalled();
});
});
describe('basics with initial active tab', function() {
beforeEach(inject(function($compile, $rootScope) {
scope = $rootScope.$new();
function makeTab(active) {
return {
active: !!active,
select: jasmine.createSpy()
};
}
scope.tabs = [
makeTab(), makeTab(), makeTab(true), makeTab()
];
elm = $compile([
'<uib-tabset>',
' <uib-tab active="tabs[0].active" select="tabs[0].select()">',
' </uib-tab>',
' <uib-tab active="tabs[1].active" select="tabs[1].select()">',
' </uib-tab>',
' <uib-tab active="tabs[2].active" select="tabs[2].select()">',
' </uib-tab>',
' <uib-tab active="tabs[3].active" select="tabs[3].select()">',
' </uib-tab>',
'</uib-tabset>'
].join('\n'))(scope);
scope.$apply();
}));
function expectTabActive(activeTab) {
var _titles = titles();
angular.forEach(scope.tabs, function(tab, i) {
if (activeTab === tab) {
expect(tab.active).toBe(true);
//It should only call select ONCE for each select
expect(tab.select).toHaveBeenCalled();
expect(_titles.eq(i)).toHaveClass('active');
expect(contents().eq(i)).toHaveClass('active');
} else {
expect(tab.active).toBe(false);
expect(_titles.eq(i)).not.toHaveClass('active');
}
});
}
it('should make tab titles and set active tab active', function() {
expect(titles().length).toBe(scope.tabs.length);
expectTabActive(scope.tabs[2]);
});
});
describe('tab callback order', function() {
var execOrder;
beforeEach(inject(function($compile, $rootScope) {
scope = $rootScope.$new();
execOrder = [];
scope.actives = {};
scope.execute = function(id) {
execOrder.push(id);
};
elm = $compile([
'<div>',
' <uib-tabset class="hello" data-pizza="pepperoni">',
' <uib-tab heading="First Tab" active="actives.one" select="execute(\'select1\')" deselect="execute(\'deselect1\')"></uib-tab>',
' <uib-tab select="execute(\'select2\')" deselect="execute(\'deselect2\')"></uib-tab>',
' </uib-tabset>',
'</div>'
].join('\n'))(scope);
scope.$apply();
return elm;
}));
it('should call select for the first tab', function() {
expect(execOrder).toEqual([ 'select1' ]);
});
it('should call deselect, then select', function() {
execOrder = [];
// Select second tab
titles().eq(1).find('a').click();
expect(execOrder).toEqual([ 'deselect1', 'select2' ]);
execOrder = [];
// Select again first tab
titles().eq(0).find('a').click();
expect(execOrder).toEqual([ 'deselect2', 'select1' ]);
});
});
describe('ng-repeat', function() {
var $compile, $rootScope;
beforeEach(inject(function(_$compile_, _$rootScope_) {
$compile = _$compile_;
$rootScope = _$rootScope_;
scope = $rootScope.$new();
scope.tabs = [
makeTab(), makeTab(), makeTab(true), makeTab()
];
elm = $compile([
'<uib-tabset>',
' <uib-tab ng-repeat="t in tabs" active="t.active" select="t.select()">',
' <uib-tab-heading><b>heading</b> {{index}}</uib-tab-heading>',
' content {{$index}}',
' </uib-tab>',
'</uib-tabset>'
].join('\n'))(scope);
scope.$apply();
}));
function makeTab(active) {
return {
active: !!active,
select: jasmine.createSpy()
};
}
function titles() {
return elm.find('ul.nav-tabs li');
}
function contents() {
return elm.find('div.tab-content div.tab-pane');
}
function expectTabActive(activeTab) {
var _titles = titles();
angular.forEach(scope.tabs, function(tab, i) {
if (activeTab === tab) {
expect(tab.active).toBe(true);
//It should only call select ONCE for each select
expect(tab.select).toHaveBeenCalled();
expect(_titles.eq(i)).toHaveClass('active');
expect(contents().eq(i).text().trim()).toBe('content ' + i);
expect(contents().eq(i)).toHaveClass('active');
} else {
expect(tab.active).toBe(false);
expect(_titles.eq(i)).not.toHaveClass('active');
}
});
}
it('should make tab titles and set active tab active', function() {
expect(titles().length).toBe(scope.tabs.length);
expectTabActive(scope.tabs[2]);
});
it('should switch active when clicking', function() {
titles().eq(3).find('a').click();
expectTabActive(scope.tabs[3]);
});
it('should switch active when setting active=true', function() {
scope.$apply('tabs[2].active = true');
expectTabActive(scope.tabs[2]);
});
it('should deselect all when no tabs are active', function() {
angular.forEach(scope.tabs, function(t) { t.active = false; });
scope.$apply();
expectTabActive(null);
expect(contents().filter('.active').length).toBe(0);
scope.tabs[2].active = true;
scope.$apply();
expectTabActive(scope.tabs[2]);
});
it('should not select twice', function() {
elm.remove();
elm = null;
scope = $rootScope.$new();
scope.tabs = [
makeTab(), makeTab(), makeTab(true), makeTab()
];
scope.foo = {active: true};
scope.select = jasmine.createSpy();
elm = $compile([
'<uib-tabset>',
' <uib-tab ng-repeat="t in tabs" active="t.active" select="select()">',
' <uib-tab-heading><b>heading</b> {{index}}</uib-tab-heading>',
' content {{$index}}',
' </uib-tab>',
' <uib-tab active="foo.active" select="select()">',
' <uib-tab-heading><b>heading</b> foo</uib-tab-heading>',
' content foo',
' </uib-tab>',
'</uib-tabset>'
].join('\n'))(scope);
scope.$apply();
expect(scope.select.calls.count()).toBe(1);
});
});
describe('advanced uib-tab-heading element', function() {
beforeEach(inject(function($compile, $rootScope, $sce) {
scope = $rootScope.$new();
scope.myHtml = $sce.trustAsHtml('<b>hello</b>, there!');
scope.value = true;
elm = $compile([
'<uib-tabset>',
' <uib-tab>',
' <uib-tab-heading ng-bind-html="myHtml" ng-show="value"></uib-tab-heading>',
' </uib-tab>',
' <uib-tab><data-uib-tab-heading>1</data-uib-tab-heading></uib-tab>',
' <uib-tab><div data-uib-tab-heading>2</div></uib-tab>',
' <uib-tab><div uib-tab-heading>3</div></uib-tab>',
'</uib-tabset>'
].join('\n'))(scope);
scope.$apply();
}));
function heading() {
return elm.find('ul li a').children();
}
it('should create a heading bound to myHtml', function() {
expect(heading().eq(0).html()).toBe('<b>hello</b>, there!');
});
it('should hide and show the heading depending on value', function() {
expect(heading().eq(0)).not.toBeHidden();
scope.$apply('value = false');
expect(heading().eq(0)).toBeHidden();
scope.$apply('value = true');
expect(heading().eq(0)).not.toBeHidden();
});
it('should have a uib-tab-heading no matter what syntax was used', function() {
expect(heading().eq(1).text()).toBe('1');
expect(heading().eq(2).text()).toBe('2');
expect(heading().eq(3).text()).toBe('3');
});
});
//Tests that http://git.io/lG6I9Q is fixed
describe('tab ordering', function() {
beforeEach(inject(function($compile, $rootScope) {
scope = $rootScope.$new();
scope.tabs = [
{ title:'Title 1', available:true },
{ title:'Title 2', available:true },
{ title:'Title 3', available:true }
];
elm = $compile([
'<uib-tabset>',
' <!-- a comment -->',
' <div>div that makes troubles</div>',
' <uib-tab heading="first">First Static</uib-tab>',
' <div>another div that may do evil</div>',
' <uib-tab ng-repeat="tab in tabs | filter:tabIsAvailable" active="tab.active" heading="{{tab.title}}">some content</uib-tab>',
' <!-- another comment -->',
' <uib-tab heading="mid">Mid Static</uib-tab>',
' a text node',
' <!-- another comment -->',
' <span>yet another span that may do evil</span>',
' <uib-tab ng-repeat="tab in tabs | filter:tabIsAvailable" active="tab.active" heading="Second {{tab.title}}">some content</uib-tab>',
' a text node',
' <span>yet another span that may do evil</span>',
' <!-- another comment -->',
' <uib-tab heading="last">Last Static</uib-tab>',
' a text node',
' <span>yet another span that may do evil</span>',
' <!-- another comment -->',
'</uib-tabset>'
].join('\n'))(scope);
scope.tabIsAvailable = function(tab) {
return tab.available;
};
}));
it('should preserve correct ordering', function() {
function titles() {
return elm.find('ul.nav-tabs li a');
}
scope.$apply();
expect(titles().length).toBe(9);
scope.$apply('tabs[1].available=false');
scope.$digest();
expect(titles().length).toBe(7);
scope.$apply('tabs[0].available=false');
scope.$digest();
expect(titles().length).toBe(5);
scope.$apply('tabs[2].available=false');
scope.$digest();
expect(titles().length).toBe(3);
scope.$apply('tabs[0].available=true');
scope.$digest();
expect(titles().length).toBe(5);
scope.$apply('tabs[1].available=true');
scope.$apply('tabs[2].available=true');
scope.$digest();
expect(titles().length).toBe(9);
expect(titles().eq(0).text().trim()).toBe('first');
expect(titles().eq(1).text().trim()).toBe('Title 1');
expect(titles().eq(2).text().trim()).toBe('Title 2');
expect(titles().eq(3).text().trim()).toBe('Title 3');
expect(titles().eq(4).text().trim()).toBe('mid');
expect(titles().eq(5).text().trim()).toBe('Second Title 1');
expect(titles().eq(6).text().trim()).toBe('Second Title 2');
expect(titles().eq(7).text().trim()).toBe('Second Title 3');
expect(titles().eq(8).text().trim()).toBe('last');
});
});
describe('uib-tabset controller', function() {
function mockTab(isActive) {
var _isActive;
if (isActive || isActive === false) {
_isActive = isActive;
}
return {
active: _isActive,
onSelect : angular.noop,
onDeselect : angular.noop
};
}
var ctrl;
beforeEach(inject(function($controller, $rootScope) {
scope = $rootScope;
//instantiate the controller stand-alone, without the directive
ctrl = $controller('UibTabsetController', {$scope: scope});
}));
describe('select', function() {
it('should mark given tab selected', function() {
var tab = mockTab();
ctrl.select(tab);
expect(tab.active).toBe(true);
});
it('should deselect other tabs', function() {
var tab1 = mockTab(), tab2 = mockTab(), tab3 = mockTab();
ctrl.addTab(tab1);
ctrl.addTab(tab2);
ctrl.addTab(tab3);
ctrl.select(tab1);
expect(tab1.active).toBe(true);
expect(tab2.active).toBe(false);
expect(tab3.active).toBe(false);
ctrl.select(tab2);
expect(tab1.active).toBe(false);
expect(tab2.active).toBe(true);
expect(tab3.active).toBe(false);
ctrl.select(tab3);
expect(tab1.active).toBe(false);
expect(tab2.active).toBe(false);
expect(tab3.active).toBe(true);
});
});
describe('addTab', function() {
it('should append tab', function() {
var tab1 = mockTab(), tab2 = mockTab();
expect(ctrl.tabs).toEqual([]);
ctrl.addTab(tab1);
expect(ctrl.tabs).toEqual([tab1]);
ctrl.addTab(tab2);
expect(ctrl.tabs).toEqual([tab1, tab2]);
});
it('should select the first one', function() {
var tab1 = mockTab(), tab2 = mockTab();
ctrl.addTab(tab1);
expect(tab1.active).toBe(true);
ctrl.addTab(tab2);
expect(tab1.active).toBe(true);
});
it('should not select first active === false tab as selected', function() {
var tab = mockTab(false);
ctrl.addTab(tab);
expect(tab.active).toBe(false);
});
it('should select a tab added that\'s already active', function() {
var tab1 = mockTab(), tab2 = mockTab(true);
ctrl.addTab(tab1);
expect(tab1.active).toBe(true);
ctrl.addTab(tab2);
expect(tab1.active).toBe(false);
expect(tab2.active).toBe(true);
});
});
});
describe('remove', function() {
it('should remove title tabs when elements are destroyed and change selection', inject(function($controller, $compile, $rootScope) {
scope = $rootScope.$new();
elm = $compile('<uib-tabset><uib-tab heading="1">Hello</uib-tab><uib-tab ng-repeat="i in list" heading="tab {{i}}">content {{i}}</uib-tab></uib-tabset>')(scope);
scope.$apply();
expectTitles(['1']);
expectContents(['Hello']);
scope.$apply('list = [1,2,3]');
expectTitles(['1', 'tab 1', 'tab 2', 'tab 3']);
expectContents(['Hello', 'content 1', 'content 2', 'content 3']);
// Select last tab
titles().find('a').eq(3).click();
expect(contents().eq(3)).toHaveClass('active');
expect(titles().eq(3)).toHaveClass('active');
// Remove last tab
scope.$apply('list = [1,2]');
expectTitles(['1', 'tab 1', 'tab 2']);
expectContents(['Hello', 'content 1', 'content 2']);
// "tab 2" is now selected
expect(titles().eq(2)).toHaveClass('active');
expect(contents().eq(2)).toHaveClass('active');
// Select 2nd tab ("tab 1")
titles().find('a').eq(1).click();
expect(titles().eq(1)).toHaveClass('active');
expect(contents().eq(1)).toHaveClass('active');
// Remove 2nd tab
scope.$apply('list = [2]');
expectTitles(['1', 'tab 2']);
expectContents(['Hello', 'content 2']);
// New 2nd tab is now selected
expect(titles().eq(1)).toHaveClass('active');
expect(contents().eq(1)).toHaveClass('active');
}));
it('should not select tabs when being destroyed', inject(function($controller, $compile, $rootScope) {
var selectList = [],
deselectList = [],
getTab = function(active) {
return {
active: active,
select : function() {
selectList.push('select');
},
deselect : function() {
deselectList.push('deselect');
}
};
};
scope = $rootScope.$new();
scope.tabs = [
getTab(true),
getTab(false)
];
elm = $compile([
'<uib-tabset>',
' <uib-tab ng-repeat="t in tabs" active="t.active" select="t.select()" deselect="t.deselect()">',
' <uib-tab-heading><b>heading</b> {{index}}</uib-tab-heading>',
' content {{$index}}',
' </uib-tab>',
'</uib-tabset>'
].join('\n'))(scope);
scope.$apply();
// The first tab is selected the during the initial $digest.
expect(selectList.length).toEqual(1);
// Destroy the tabs - we should not trigger selection/deselection any more.
scope.$destroy();
expect(selectList.length).toEqual(1);
expect(deselectList.length).toEqual(0);
}));
});
describe('disable', function() {
beforeEach(inject(function($compile, $rootScope) {
scope = $rootScope.$new();
function makeTab(disable) {
return {
active: false,
select: jasmine.createSpy(),
disable: disable
};
}
scope.tabs = [
makeTab(false), makeTab(true), makeTab(false), makeTab(true)
];
elm = $compile([
'<uib-tabset>',
' <uib-tab ng-repeat="t in tabs" active="t.active" select="t.select()" disable="t.disable">',
' <uib-tab-heading><b>heading</b> {{index}}</uib-tab-heading>',
' content {{$index}}',
' </uib-tab>',
'</uib-tabset>'
].join('\n'))(scope);
scope.$apply();
}));
function expectTabActive(activeTab) {
var _titles = titles();
angular.forEach(scope.tabs, function(tab, i) {
if (activeTab === tab) {
expect(tab.active).toBe(true);
expect(tab.select.calls.count()).toBe( (tab.disable) ? 0 : 1 );
expect(_titles.eq(i)).toHaveClass('active');
expect(contents().eq(i).text().trim()).toBe('content ' + i);
expect(contents().eq(i)).toHaveClass('active');
} else {
expect(tab.active).toBe(false);
expect(_titles.eq(i)).not.toHaveClass('active');
}
});
}
it('should not switch active when clicking on title', function() {
titles().eq(2).find('a').click();
expectTabActive(scope.tabs[2]);
titles().eq(3).find('a').click();
expectTabActive(scope.tabs[2]);
});
it('should toggle between states', function() {
expect(titles().eq(3)).toHaveClass('disabled');
scope.$apply('tabs[3].disable = false');
expect(titles().eq(3)).not.toHaveClass('disabled');
expect(titles().eq(2)).not.toHaveClass('disabled');
scope.$apply('tabs[2].disable = true');
expect(titles().eq(2)).toHaveClass('disabled');
});
});
describe('vertical', function() {
beforeEach(inject(function($compile, $rootScope) {
scope = $rootScope.$new();
scope.vertical = true;
elm = $compile('<uib-tabset vertical="vertical"></uib-tabset>')(scope);
scope.$apply();
}));
it('to stack tabs', function() {
expect(elm.find('ul.nav-tabs')).toHaveClass('nav-stacked');
});
});
describe('justified', function() {
beforeEach(inject(function($compile, $rootScope) {
scope = $rootScope.$new();
scope.justified = true;
elm = $compile('<uib-tabset justified="justified"></uib-tabset>')(scope);
scope.$apply();
}));
it('to justify tabs', function() {
expect(elm.find('ul.nav-tabs')).toHaveClass('nav-justified');
});
});
describe('type', function() {
beforeEach(inject(function($compile, $rootScope) {
scope = $rootScope.$new();
scope.navType = 'pills';
elm = $compile('<uib-tabset type="{{navType}}"></uib-tabset>')(scope);
scope.$apply();
}));
it('to show pills', function() {
expect(elm.find('ul')).toHaveClass('nav-pills');
expect(elm.find('ul')).not.toHaveClass('nav-tabs');
});
});
//https://github.com/angular-ui/bootstrap/issues/524
describe('child compilation', function() {
var elm;
beforeEach(inject(function($compile, $rootScope) {
elm = $compile('<uib-tabset><uib-tab><div></div></uib-tab></uib-tabset></div>')($rootScope.$new());
$rootScope.$apply();
}));
it('should hookup the tab\'s children to the tab with $compile', function() {
var tabChild = $('.tab-pane', elm).children().first();
expect(tabChild.inheritedData('$uibTabsetController')).toBeTruthy();
});
});
//https://github.com/angular-ui/bootstrap/issues/631
describe('ng-options in content', function() {
var elm;
it('should render correct amount of options', inject(function($compile, $rootScope) {
var scope = $rootScope.$new();
elm = $compile('<uib-tabset><uib-tab><select ng-model="foo" ng-options="i for i in [1,2,3]"></uib-tab></uib-tabset>')(scope);
scope.$apply();
var select = elm.find('select');
scope.$apply();
expect(select.children().length).toBe(4);
}));
});
//https://github.com/angular-ui/bootstrap/issues/599
describe('ng-repeat in content', function() {
var elm;
it('should render ng-repeat', inject(function($compile, $rootScope) {
var scope = $rootScope.$new();
scope.tabs = [
{title:'a', array:[1,2,3]},
{title:'b', array:[2,3,4]},
{title:'c', array:[3,4,5]}
];
elm = $compile('<div><uib-tabset>' +
'<uib-tab ng-repeat="tab in tabs" heading="{{tab.title}}">' +
'<uib-tab-heading>{{$index}}</uib-tab-heading>' +
'<span ng-repeat="a in tab.array">{{a}},</span>' +
'</uib-tab>' +
'</uib-tabset></div>')(scope);
scope.$apply();
var contents = elm.find('.tab-pane');
expect(contents.eq(0).text().trim()).toEqual('1,2,3,');
expect(contents.eq(1).text().trim()).toEqual('2,3,4,');
expect(contents.eq(2).text().trim()).toEqual('3,4,5,');
}));
});
//https://github.com/angular-ui/bootstrap/issues/783
describe('nested tabs', function() {
var elm;
it('should render without errors', inject(function($compile, $rootScope) {
var scope = $rootScope.$new();
elm = $compile([
'<div>',
' <uib-tabset class="tabbable">',
' <uib-tab heading="Tab 1">',
' <uib-tabset class="tabbable">',
' <uib-tab heading="Tab 1A">',
' </uib-tab>',
' </uib-tabset>',
' </uib-tab>',
' <uib-tab heading="Tab 2">',
' <uib-tabset class="tabbable">',
' <uib-tab heading="Tab 2A">',
' </uib-tab>',
' </uib-tabset>',
' </uib-tab>',
' </uib-tabset>',
'</div>'
].join('\n'))(scope);
scope.$apply();
// 1 outside tabset, 2 nested tabsets
expect(elm.find('.tabbable').length).toEqual(3);
}));
it('should render with the correct scopes', inject(function($compile, $rootScope) {
var scope = $rootScope.$new();
scope.tab1Text = 'abc';
scope.tab1aText = '123';
scope.tab1aHead = '123';
scope.tab2aaText = '456';
elm = $compile([
'<div>',
' <uib-tabset class="tabbable">',
' <uib-tab heading="Tab 1">',
' <uib-tabset class="tabbable">',
' <uib-tab heading="{{ tab1aHead }}">',
' {{ tab1aText }}',
' </uib-tab>',
' </uib-tabset>',
' <span class="tab-1">{{ tab1Text }}</span>',
' </uib-tab>',
' <uib-tab heading="Tab 2">',
' <uib-tabset class="tabbable">',
' <uib-tab heading="Tab 2A">',
' <uib-tabset class="tabbable">',
' <uib-tab heading="Tab 2AA">',
' <span class="tab-2aa">{{ tab2aaText }}</span>',
' </uib-tab>',
' </uib-tabset>',
' </uib-tab>',
' </uib-tabset>',
' </uib-tab>',
' </uib-tabset>',
'</div>'
].join('\n'))(scope);
scope.$apply();
var outsideTabset = elm.find('.tabbable').eq(0);
var nestedTabset = outsideTabset.find('.tabbable');
expect(elm.find('.tabbable').length).toEqual(4);
expect(outsideTabset.find('.tab-pane').eq(0).find('.tab-1').text().trim()).toEqual(scope.tab1Text);
expect(nestedTabset.find('.tab-pane').eq(0).text().trim()).toEqual(scope.tab1aText);
expect(nestedTabset.find('ul.nav-tabs li').eq(0).text().trim()).toEqual(scope.tab1aHead);
expect(nestedTabset.eq(2).find('.tab-pane').eq(0).find('.tab-2aa').text().trim()).toEqual(scope.tab2aaText);
}));
it('ng-repeat works with nested tabs', inject(function($compile, $rootScope) {
var scope = $rootScope.$new();
scope.tabs = [
{
tabs: [
{
content: 'tab1a'
},
{
content: 'tab2a'
}
],
content: 'tab1'
}
];
elm = $compile([
'<div>',
' <uib-tabset>',
' <uib-tab ng-repeat="tab in tabs">',
' <uib-tabset>',
' <uib-tab ng-repeat="innerTab in tab.tabs">',
' <span class="inner-tab-content">{{ innerTab.content }}</span>',
' </uib-tab>',
' </uib-tabset>',
' <span class="outer-tab-content">{{ tab.content }}</span>',
' </uib-tab>',
' </uib-tabset>',
'</div>'
].join('\n'))(scope);
scope.$apply();
expect(elm.find('.inner-tab-content').eq(0).text().trim()).toEqual(scope.tabs[0].tabs[0].content);
expect(elm.find('.inner-tab-content').eq(1).text().trim()).toEqual(scope.tabs[0].tabs[1].content);
expect(elm.find('.outer-tab-content').eq(0).text().trim()).toEqual(scope.tabs[0].content);
}));
});
});
/* deprecation tests below */
describe('tab deprecation', function() {
beforeEach(module('ui.bootstrap.tabs'));
beforeEach(module('template/tabs/tabset.html'));
beforeEach(module('template/tabs/tab.html'));
it('should suppress warning', function() {
module(function($provide) {
$provide.value('$tabsSuppressWarning', true);
});
inject(function($compile, $log, $rootScope) {
spyOn($log, 'warn');
var element =
'<tabset>' +
'<tab>' +
'<tab-heading>Tab Heading One</tab-heading>' +
'<div>Tab One Contents</div>' +
'</tab>' +
'<tab heading="Tab Heading Two">' +
'<div>Tab Two Contents</div>' +
'</tab>' +
'</tabset>';
$compile(element)($rootScope);
$rootScope.$digest();
expect($log.warn.calls.count()).toBe(0);
});
});
it('should give warning by default', inject(function($templateCache, $compile, $log, $rootScope) {
spyOn($log, 'warn');
var tabSetTemplate =
'<div>' +
'<ul class="nav nav-{{type || \'tabs\'}}" ng-class="{\'nav-stacked\': vertical, \'nav-justified\': justified}" ng-transclude></ul>' +
'<div class="tab-content">' +
'<div class="tab-pane" ng-repeat="tab in tabs" ng-class="{active: tab.active}" tab-content-transclude="tab"></div>' +
'</div>' +
'</div>';
$templateCache.put('template/tabs/tabset.html', tabSetTemplate);
var tabTemplate =
'<li ng-class="{active: active, disabled: disabled}">' +
'<a href ng-click="select()" tab-heading-transclude>{{heading}}</a>' +
'</li>';
$templateCache.put('template/tabs/tab.html', tabTemplate);
var element =
'<tabset>' +
'<tab>' +
'<tab-heading>Tab Heading One</tab-heading>' +
'<div>Tab One Contents</div>' +
'</tab>' +
'</tabset>';
$compile(element)($rootScope);
$rootScope.$digest();
expect($log.warn.calls.count()).toBe(5);
expect($log.warn.calls.argsFor(0)).toEqual(['TabsetController is now deprecated. Use UibTabsetController instead.']);
expect($log.warn.calls.argsFor(1)).toEqual(['tab-heading-transclude is now deprecated. Use uib-tab-heading-transclude instead.']);
expect($log.warn.calls.argsFor(2)).toEqual(['tab is now deprecated. Use uib-tab instead.']);
expect($log.warn.calls.argsFor(3)).toEqual(['tabset is now deprecated. Use uib-tabset instead.']);
expect($log.warn.calls.argsFor(4)).toEqual(['tab-content-transclude is now deprecated. Use uib-tab-content-transclude instead.']);
}));
});