Skip to content

Commit

Permalink
feat(pagination): option for different mode when maxSize
Browse files Browse the repository at this point in the history
 * Allow user to specify if current page is rotated, moving the left/right
   limit, or page is moving between page groups.
  • Loading branch information
bekos authored and ajoslin committed May 22, 2013
1 parent b0745c8 commit a023d08
Show file tree
Hide file tree
Showing 5 changed files with 132 additions and 17 deletions.
18 changes: 13 additions & 5 deletions src/pagination/docs/demo.html
Original file line number Diff line number Diff line change
@@ -1,9 +1,17 @@
<div ng-controller="PaginationDemoCtrl">
<div ng-controller="PaginationDemoCtrl" class="well well-small">
<h4>Default</h4>

<pagination num-pages="noOfPages" current-page="currentPage"></pagination>
<pagination num-pages="noOfPages" current-page="currentPage" class="pagination-small" previous-text="&laquo;" next-text="&raquo;"></pagination>
<pagination boundary-links="true" num-pages="noOfPages" current-page="currentPage" max-size="maxSize"></pagination>
<pagination num-pages="noOfPages" current-page="currentPage" max-size="maxSize"></pagination>
<pagination boundary-links="true" num-pages="noOfPages" current-page="currentPage" class="pagination-small" previous-text="&lsaquo;" next-text="&rsaquo;" first-text="&laquo;" last-text="&raquo;"></pagination>
<pagination direction-links="false" boundary-links="true" num-pages="noOfPages" current-page="currentPage"></pagination>
<pagination direction-links="false" num-pages="noOfPages" current-page="currentPage"></pagination>

<button class="btn" ng-click="setPage(3)">Set current page to: 3</button>
The selected page no: {{currentPage}}
</div>

<hr />

<h4>Limit the maximimum visible page-buttons</h4>
<pagination num-pages="bigNoOfPages" current-page="bigCurrentPage" max-size="maxSize" class="pagination-mini" boundary-links="true"></pagination>
<pagination rotate="false" num-pages="bigNoOfPages" current-page="bigCurrentPage" max-size="maxSize" class="pagination-mini" boundary-links="true"></pagination>
</div>
3 changes: 3 additions & 0 deletions src/pagination/docs/demo.js
Original file line number Diff line number Diff line change
Expand Up @@ -6,4 +6,7 @@ var PaginationDemoCtrl = function ($scope) {
$scope.setPage = function (pageNo) {
$scope.currentPage = pageNo;
};

$scope.bigNoOfPages = 18;
$scope.bigCurrentPage = 1;
};
3 changes: 2 additions & 1 deletion src/pagination/docs/readme.md
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@

A lightweight pagination directive that is focused on ... providing pagination!

It will take care of visualising a pagination bar. Additionally it will make sure that the state (enabled / disabled) of the Previous / Next and First / Last buttons (if exist) is maintained correctly.

It also provides optional attribute max-size to limit the size of pagination bar.
It also provides optional attribute max-size to limit the size of pagination bar & rotate attribute whether to keep current page in the middle of the visible ones.
46 changes: 35 additions & 11 deletions src/pagination/pagination.js
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,8 @@ angular.module('ui.bootstrap.pagination', [])
firstText: 'First',
previousText: 'Previous',
nextText: 'Next',
lastText: 'Last'
lastText: 'Last',
rotate: true
})

.directive('pagination', ['paginationConfig', function(paginationConfig) {
Expand All @@ -29,6 +30,7 @@ angular.module('ui.bootstrap.pagination', [])
var previousText = angular.isDefined(attrs.previousText) ? attrs.previousText : paginationConfig.previousText;
var nextText = angular.isDefined(attrs.nextText) ? attrs.nextText : paginationConfig.nextText;
var lastText = angular.isDefined(attrs.lastText) ? attrs.lastText : paginationConfig.lastText;
var rotate = angular.isDefined(attrs.rotate) ? scope.$eval(attrs.rotate) : paginationConfig.rotate;

// Create page object used in template
function makePage(number, text, isActive, isDisabled) {
Expand All @@ -45,16 +47,26 @@ angular.module('ui.bootstrap.pagination', [])

// Default page limits
var startPage = 1, endPage = scope.numPages;
var isMaxSized = ( angular.isDefined(scope.maxSize) && scope.maxSize < scope.numPages );

// recompute if maxSize
if ( scope.maxSize && scope.maxSize < scope.numPages ) {
startPage = Math.max(scope.currentPage - Math.floor(scope.maxSize/2), 1);
endPage = startPage + scope.maxSize - 1;

// Adjust if limit is exceeded
if (endPage > scope.numPages) {
endPage = scope.numPages;
startPage = endPage - scope.maxSize + 1;
if ( isMaxSized ) {
if ( rotate ) {
// Current page is displayed in the middle of the visible ones
startPage = Math.max(scope.currentPage - Math.floor(scope.maxSize/2), 1);
endPage = startPage + scope.maxSize - 1;

// Adjust if limit is exceeded
if (endPage > scope.numPages) {
endPage = scope.numPages;
startPage = endPage - scope.maxSize + 1;
}
} else {
// Visible pages are paginated with maxSize
startPage = ((Math.ceil(scope.currentPage / scope.maxSize) - 1) * scope.maxSize) + 1;

// Adjust last page if limit is exceeded
endPage = Math.min(startPage + scope.maxSize - 1, scope.numPages);
}
}

Expand All @@ -64,6 +76,19 @@ angular.module('ui.bootstrap.pagination', [])
scope.pages.push(page);
}

// Add links to move between page sets
if ( isMaxSized && ! rotate ) {
if ( startPage > 1 ) {
var previousPageSet = makePage(startPage - 1, '...', false, false);
scope.pages.unshift(previousPageSet);
}

if ( endPage < scope.numPages ) {
var nextPageSet = makePage(endPage + 1, '...', false, false);
scope.pages.push(nextPageSet);
}
}

// Add previous & next links
if (directionLinks) {
var previousPage = makePage(scope.currentPage - 1, previousText, false, scope.noPrevious());
Expand All @@ -82,7 +107,6 @@ angular.module('ui.bootstrap.pagination', [])
scope.pages.push(lastPage);
}


if ( scope.currentPage > scope.numPages ) {
scope.selectPage(scope.numPages);
}
Expand All @@ -105,4 +129,4 @@ angular.module('ui.bootstrap.pagination', [])
};
}
};
}]);
}]);
79 changes: 79 additions & 0 deletions src/pagination/test/pagination.spec.js
Original file line number Diff line number Diff line change
Expand Up @@ -190,6 +190,85 @@ describe('pagination directive with max size option', function () {
expect($rootScope.maxSize).toBe(15);
});

it('should not display page numbers, if max-size is zero', function() {
$rootScope.maxSize = 0;
$rootScope.$digest();
expect(element.find('li').length).toBe(2);
expect(element.find('li').eq(0).text()).toBe('Previous');
expect(element.find('li').eq(-1).text()).toBe('Next');
});

});

describe('pagination directive with max size option & no rotate', function () {
var $rootScope, element;
beforeEach(module('ui.bootstrap.pagination'));
beforeEach(module('template/pagination/pagination.html'));
beforeEach(inject(function(_$compile_, _$rootScope_) {
$compile = _$compile_;
$rootScope = _$rootScope_;
$rootScope.numPages = 12;
$rootScope.currentPage = 7;
$rootScope.maxSize = 5;
element = $compile('<pagination num-pages="numPages" current-page="currentPage" max-size="maxSize" rotate="false"></pagination>')($rootScope);
$rootScope.$digest();
}));

it('contains one ul and maxsize + 4 elements', function() {
expect(element.find('ul').length).toBe(1);
expect(element.find('li').length).toBe($rootScope.maxSize + 4);
expect(element.find('li').eq(0).text()).toBe('Previous');
expect(element.find('li').eq(1).text()).toBe('...');
expect(element.find('li').eq(2).text()).toBe('6');
expect(element.find('li').eq(-3).text()).toBe('10');
expect(element.find('li').eq(-2).text()).toBe('...');
expect(element.find('li').eq(-1).text()).toBe('Next');
});

it('shows only the next ellipsis element on first page set', function() {
$rootScope.currentPage = 3;
$rootScope.$digest();
expect(element.find('li').eq(1).text()).toBe('1');
expect(element.find('li').eq(-3).text()).toBe('5');
expect(element.find('li').eq(-2).text()).toBe('...');
});

it('shows only the previous ellipsis element on last page set', function() {
$rootScope.currentPage = 12;
$rootScope.$digest();
expect(element.find('li').length).toBe(5);
expect(element.find('li').eq(1).text()).toBe('...');
expect(element.find('li').eq(2).text()).toBe('11');
expect(element.find('li').eq(-2).text()).toBe('12');
});

it('moves to the previous set when first ellipsis is clicked', function() {
var prev = element.find('li').eq(1).find('a').eq(0);
expect(prev.text()).toBe('...');

prev.click();
expect($rootScope.currentPage).toBe(5);
var currentPageItem = element.find('li').eq(-3);
expect(currentPageItem.hasClass('active')).toBe(true);
});

it('moves to the next set when last ellipsis is clicked', function() {
var next = element.find('li').eq(-2).find('a').eq(0);
expect(next.text()).toBe('...');

next.click();
expect($rootScope.currentPage).toBe(11);
var currentPageItem = element.find('li').eq(2);
expect(currentPageItem.hasClass('active')).toBe(true);
});

it('should not display page numbers, if max-size is zero', function() {
$rootScope.maxSize = 0;
$rootScope.$digest();
expect(element.find('li').length).toBe(2);
expect(element.find('li').eq(0).text()).toBe('Previous');
expect(element.find('li').eq(-1).text()).toBe('Next');
});
});

describe('pagination directive with added first & last links', function () {
Expand Down

0 comments on commit a023d08

Please sign in to comment.