2018-03-19 02:16:27 +00:00
2018-03-05 11:06:44 +00:00
import PropTypes from '../_util/vue-types'
import BaseMixin from '../_util/BaseMixin'
2018-11-08 11:59:09 +00:00
import { hasProp , getComponentFromProp } from '../_util/props-util'
2018-03-05 11:06:44 +00:00
import Pager from './Pager'
import Options from './Options'
import LOCALE from './locale/zh_CN'
import KEYCODE from './KeyCode'
function noop ( ) {
}
// 是否是正整数
function isInteger ( value ) {
return typeof value === 'number' &&
isFinite ( value ) &&
Math . floor ( value ) === value
}
function defaultItemRender ( page , type , element ) {
return element
}
2019-01-01 13:19:50 +00:00
function calculatePage ( p , state , props ) {
let pageSize = p
if ( typeof pageSize === 'undefined' ) {
pageSize = state . statePageSize
}
return Math . floor ( ( props . total - 1 ) / pageSize ) + 1
}
2018-03-05 11:06:44 +00:00
export default {
name : 'Pagination' ,
mixins : [ BaseMixin ] ,
props : {
prefixCls : PropTypes . string . def ( 'rc-pagination' ) ,
selectPrefixCls : PropTypes . string . def ( 'rc-select' ) ,
current : PropTypes . number ,
defaultCurrent : PropTypes . number . def ( 1 ) ,
total : PropTypes . number . def ( 0 ) ,
pageSize : PropTypes . number ,
defaultPageSize : PropTypes . number . def ( 10 ) ,
hideOnSinglePage : PropTypes . bool . def ( false ) ,
showSizeChanger : PropTypes . bool . def ( false ) ,
showLessItems : PropTypes . bool . def ( false ) ,
// showSizeChange: PropTypes.func.def(noop),
selectComponentClass : PropTypes . any ,
2018-03-22 09:23:51 +00:00
showPrevNextJumpers : PropTypes . bool . def ( true ) ,
2018-03-05 11:06:44 +00:00
showQuickJumper : PropTypes . oneOfType ( [ PropTypes . bool , PropTypes . object ] ) . def ( false ) ,
showTitle : PropTypes . bool . def ( true ) ,
pageSizeOptions : PropTypes . arrayOf ( PropTypes . string ) ,
2018-03-06 07:40:42 +00:00
buildOptionText : PropTypes . func ,
2018-03-05 11:06:44 +00:00
showTotal : PropTypes . func ,
simple : PropTypes . bool ,
locale : PropTypes . object . def ( LOCALE ) ,
itemRender : PropTypes . func . def ( defaultItemRender ) ,
2018-11-08 11:59:09 +00:00
prevIcon : PropTypes . any ,
nextIcon : PropTypes . any ,
jumpPrevIcon : PropTypes . any ,
jumpNextIcon : PropTypes . any ,
2018-03-05 11:06:44 +00:00
} ,
model : {
prop : 'current' ,
2018-10-23 04:49:40 +00:00
event : 'change.current' ,
2018-03-05 11:06:44 +00:00
} ,
data ( ) {
const hasOnChange = this . onChange !== noop
const hasCurrent = hasProp ( this , 'current' )
if ( hasCurrent && ! hasOnChange ) {
console . warn ( 'Warning: You provided a `current` prop to a Pagination component without an `onChange` handler. This will render a read-only component.' ) ; // eslint-disable-line
}
let current = this . defaultCurrent
if ( hasCurrent ) {
current = this . current
}
let pageSize = this . defaultPageSize
if ( hasProp ( this , 'pageSize' ) ) {
pageSize = this . pageSize
}
return {
stateCurrent : current ,
stateCurrentInputValue : current ,
statePageSize : pageSize ,
}
} ,
watch : {
current ( val ) {
this . setState ( {
stateCurrent : val ,
stateCurrentInputValue : val ,
} )
} ,
pageSize ( val ) {
const newState = { }
2018-03-11 05:47:42 +00:00
let current = this . stateCurrent
2019-01-01 13:19:50 +00:00
const newCurrent = calculatePage ( val , this . $data , this . $props )
2018-03-05 11:06:44 +00:00
current = current > newCurrent ? newCurrent : current
2018-03-07 02:48:33 +00:00
if ( ! hasProp ( this , 'current' ) ) {
2018-03-05 11:06:44 +00:00
newState . stateCurrent = current
newState . stateCurrentInputValue = current
}
newState . statePageSize = val
this . setState ( newState )
} ,
2018-04-21 04:51:56 +00:00
stateCurrent ( val , oldValue ) {
2018-04-07 10:52:02 +00:00
// When current page change, fix focused style of prev item
// A hacky solution of https://github.com/ant-design/ant-design/issues/8948
2018-04-21 04:51:56 +00:00
this . $nextTick ( ( ) => {
if ( this . $refs . paginationNode ) {
2018-04-07 10:52:02 +00:00
const lastCurrentNode = this . $refs . paginationNode . querySelector (
` . ${ this . prefixCls } -item- ${ oldValue } `
2018-04-21 04:51:56 +00:00
)
2018-04-07 10:52:02 +00:00
if ( lastCurrentNode && document . activeElement === lastCurrentNode ) {
2018-04-21 04:51:56 +00:00
lastCurrentNode . blur ( )
2018-04-07 10:52:02 +00:00
}
}
} )
} ,
2018-03-05 11:06:44 +00:00
} ,
methods : {
2019-01-01 13:19:50 +00:00
getJumpPrevPage ( ) {
return Math . max ( 1 , this . stateCurrent - ( this . showLessItems ? 3 : 5 ) )
} ,
getJumpNextPage ( ) {
return Math . min (
calculatePage ( undefined , this . $data , this . $props ) ,
this . stateCurrent + ( this . showLessItems ? 3 : 5 )
)
2018-03-05 11:06:44 +00:00
} ,
2018-11-08 11:59:09 +00:00
getItemIcon ( icon ) {
const { prefixCls } = this . $props
const iconNode = getComponentFromProp ( this , icon , this . $props ) || < a class = { ` ${ prefixCls } -item-link ` } / >
return iconNode
} ,
2019-01-01 13:19:50 +00:00
isValid ( page ) {
return isInteger ( page ) && page >= 1 && page !== this . stateCurrent
2018-03-05 11:06:44 +00:00
} ,
2019-01-01 13:19:50 +00:00
// calculatePage (p) {
// let pageSize = p
// if (typeof pageSize === 'undefined') {
// pageSize = this.statePageSize
// }
// return Math.floor((this.total - 1) / pageSize) + 1
// },
2018-03-05 11:06:44 +00:00
handleKeyDown ( event ) {
if ( event . keyCode === KEYCODE . ARROW _UP || event . keyCode === KEYCODE . ARROW _DOWN ) {
event . preventDefault ( )
}
} ,
handleKeyUp ( event ) {
const inputValue = event . target . value
const stateCurrentInputValue = this . stateCurrentInputValue
let value
if ( inputValue === '' ) {
value = inputValue
} else if ( isNaN ( Number ( inputValue ) ) ) {
value = stateCurrentInputValue
} else {
value = Number ( inputValue )
}
if ( value !== stateCurrentInputValue ) {
this . setState ( {
stateCurrentInputValue : value ,
} )
}
if ( event . keyCode === KEYCODE . ENTER ) {
this . handleChange ( value )
} else if ( event . keyCode === KEYCODE . ARROW _UP ) {
this . handleChange ( value - 1 )
} else if ( event . keyCode === KEYCODE . ARROW _DOWN ) {
this . handleChange ( value + 1 )
}
} ,
changePageSize ( size ) {
let current = this . stateCurrent
2018-10-23 04:49:40 +00:00
const preCurrent = current
2019-01-01 13:19:50 +00:00
const newCurrent = calculatePage ( size , this . $data , this . $props )
2018-03-05 11:06:44 +00:00
current = current > newCurrent ? newCurrent : current
2018-09-05 13:28:54 +00:00
// fix the issue:
// Once 'total' is 0, 'current' in 'onShowSizeChange' is 0, which is not correct.
if ( newCurrent === 0 ) {
current = this . stateCurrent
}
2018-03-05 11:06:44 +00:00
if ( typeof size === 'number' ) {
if ( ! hasProp ( this , 'pageSize' ) ) {
this . setState ( {
statePageSize : size ,
} )
}
if ( ! hasProp ( this , 'current' ) ) {
this . setState ( {
stateCurrent : current ,
stateCurrentInputValue : current ,
} )
}
}
2018-03-11 05:47:42 +00:00
this . $emit ( 'update:pageSize' , size )
2018-03-05 11:06:44 +00:00
this . $emit ( 'showSizeChange' , current , size )
2018-10-23 04:49:40 +00:00
if ( current !== preCurrent ) {
this . $emit ( 'change.current' , current , size )
}
2018-03-05 11:06:44 +00:00
} ,
handleChange ( p ) {
let page = p
if ( this . isValid ( page ) ) {
2019-01-01 13:19:50 +00:00
const currentPage = calculatePage ( undefined , this . $data , this . $props )
if ( page > currentPage ) {
page = currentPage
2018-03-05 11:06:44 +00:00
}
if ( ! hasProp ( this , 'current' ) ) {
this . setState ( {
stateCurrent : page ,
stateCurrentInputValue : page ,
} )
}
2018-03-07 02:48:33 +00:00
// this.$emit('input', page)
2018-03-05 11:06:44 +00:00
this . $emit ( 'change' , page , this . statePageSize )
2018-10-23 04:49:40 +00:00
this . $emit ( 'change.current' , page , this . statePageSize )
2018-03-05 11:06:44 +00:00
return page
}
return this . stateCurrent
} ,
2019-01-01 13:19:50 +00:00
prev ( ) {
if ( this . hasPrev ( ) ) {
this . handleChange ( this . stateCurrent - 1 )
}
} ,
next ( ) {
if ( this . hasNext ( ) ) {
this . handleChange ( this . stateCurrent + 1 )
}
} ,
jumpPrev ( ) {
this . handleChange ( this . getJumpPrevPage ( ) )
} ,
jumpNext ( ) {
this . handleChange ( this . getJumpNextPage ( ) )
} ,
hasPrev ( ) {
return this . stateCurrent > 1
} ,
hasNext ( ) {
return this . stateCurrent < calculatePage ( undefined , this . $data , this . $props )
} ,
2018-03-05 11:06:44 +00:00
runIfEnter ( event , callback , ... restParams ) {
if ( event . key === 'Enter' || event . charCode === 13 ) {
callback ( ... restParams )
}
} ,
runIfEnterPrev ( event ) {
this . runIfEnter ( event , this . prev )
} ,
runIfEnterNext ( event ) {
this . runIfEnter ( event , this . next )
} ,
runIfEnterJumpPrev ( event ) {
this . runIfEnter ( event , this . jumpPrev )
} ,
runIfEnterJumpNext ( event ) {
this . runIfEnter ( event , this . jumpNext )
} ,
2019-01-01 13:19:50 +00:00
handleGoTO ( event ) {
if ( event . keyCode === KEYCODE . ENTER || event . type === 'click' ) {
this . handleChange ( this . stateCurrentInputValue )
}
2018-03-05 11:06:44 +00:00
} ,
} ,
render ( ) {
// When hideOnSinglePage is true and there is only 1 page, hide the pager
if ( this . hideOnSinglePage === true && this . total <= this . statePageSize ) {
return null
}
2018-11-08 11:59:09 +00:00
const props = this . $props
2018-03-05 11:06:44 +00:00
const locale = this . locale
const prefixCls = this . prefixCls
2019-01-01 13:19:50 +00:00
const allPages = calculatePage ( undefined , this . $data , this . $props )
2018-03-05 11:06:44 +00:00
const pagerList = [ ]
let jumpPrev = null
let jumpNext = null
let firstPager = null
let lastPager = null
let gotoButton = null
const goButton = ( this . showQuickJumper && this . showQuickJumper . goButton )
const pageBufferSize = this . showLessItems ? 1 : 2
const { stateCurrent , statePageSize } = this
const prevPage = stateCurrent - 1 > 0 ? stateCurrent - 1 : 0
const nextPage = stateCurrent + 1 < allPages ? stateCurrent + 1 : allPages
if ( this . simple ) {
if ( goButton ) {
if ( typeof goButton === 'boolean' ) {
gotoButton = (
< button
type = 'button'
onClick = { this . handleGoTO }
2018-03-07 02:48:33 +00:00
onKeyup = { this . handleGoTO }
2018-03-05 11:06:44 +00:00
>
{ locale . jump _to _confirm }
< / button >
)
} else {
gotoButton = (
< span
onClick = { this . handleGoTO }
2018-03-07 02:48:33 +00:00
onKeyup = { this . handleGoTO }
2018-03-05 11:06:44 +00:00
> { goButton } < / span >
)
}
gotoButton = (
< li
title = { this . showTitle ? ` ${ locale . jump _to } ${ this . stateCurrent } / ${ allPages } ` : null }
class = { ` ${ prefixCls } -simple-pager ` }
>
{ gotoButton }
< / li >
)
}
2018-03-09 04:20:21 +00:00
const hasPrev = this . hasPrev ( )
const hasNext = this . hasNext ( )
2018-03-05 11:06:44 +00:00
return (
< ul class = { ` ${ prefixCls } ${ prefixCls } -simple ` } >
< li
title = { this . showTitle ? locale . prev _page : null }
onClick = { this . prev }
2018-03-09 04:20:21 +00:00
tabIndex = { hasPrev ? 0 : null }
2018-03-05 11:06:44 +00:00
onKeypress = { this . runIfEnterPrev }
2018-03-09 04:20:21 +00:00
class = { ` ${ hasPrev ? '' : ` ${ prefixCls } -disabled ` } ${ prefixCls } -prev ` }
2018-03-05 11:06:44 +00:00
aria - disabled = { ! this . hasPrev ( ) }
>
2018-11-08 11:59:09 +00:00
{ this . itemRender ( prevPage , 'prev' , this . getItemIcon ( 'prevIcon' ) ) }
2018-03-05 11:06:44 +00:00
< / li >
< li
title = { this . showTitle ? ` ${ stateCurrent } / ${ allPages } ` : null }
class = { ` ${ prefixCls } -simple-pager ` }
>
< input
type = 'text'
value = { this . stateCurrentInputValue }
onKeydown = { this . handleKeyDown }
onKeyup = { this . handleKeyUp }
2018-03-12 14:13:59 +00:00
onInput = { this . handleKeyUp }
2018-03-05 11:06:44 +00:00
size = '3'
/ >
< span class = { ` ${ prefixCls } -slash ` } > / < / span >
{ allPages }
< / li >
< li
title = { this . showTitle ? locale . next _page : null }
onClick = { this . next }
2018-03-22 09:23:51 +00:00
tabIndex = { this . hasNext ? 0 : null }
2018-03-05 11:06:44 +00:00
onKeypress = { this . runIfEnterNext }
2018-03-09 04:20:21 +00:00
class = { ` ${ hasNext ? '' : ` ${ prefixCls } -disabled ` } ${ prefixCls } -next ` }
2018-03-05 11:06:44 +00:00
aria - disabled = { ! this . hasNext ( ) }
>
2018-11-08 11:59:09 +00:00
{ this . itemRender ( nextPage , 'next' , this . getItemIcon ( 'nextIcon' ) ) }
2018-03-05 11:06:44 +00:00
< / li >
{ gotoButton }
< / ul >
)
}
if ( allPages <= 5 + pageBufferSize * 2 ) {
2019-01-01 13:19:50 +00:00
const pagerProps = {
props : {
locale ,
rootPrefixCls : prefixCls ,
showTitle : props . showTitle ,
itemRender : props . itemRender ,
} ,
on : {
click : this . handleChange ,
keypress : this . runIfEnter ,
} ,
}
if ( ! allPages ) {
pagerList . push (
< Pager
{ ... pagerProps }
key = 'noPager'
page = { allPages }
class = { ` ${ prefixCls } -disabled ` }
/ >
)
}
2018-03-05 11:06:44 +00:00
for ( let i = 1 ; i <= allPages ; i ++ ) {
const active = stateCurrent === i
pagerList . push (
< Pager
2019-01-01 13:19:50 +00:00
{ ... pagerProps }
2018-03-05 11:06:44 +00:00
key = { i }
page = { i }
active = { active }
/ >
)
}
} else {
const prevItemTitle = this . showLessItems ? locale . prev _3 : locale . prev _5
const nextItemTitle = this . showLessItems ? locale . next _3 : locale . next _5
2018-03-22 09:23:51 +00:00
if ( this . showPrevNextJumpers ) {
2018-11-08 11:59:09 +00:00
let jumpPrevClassString = ` ${ prefixCls } -jump-prev `
if ( props . jumpPrevIcon ) {
jumpPrevClassString += ` ${ prefixCls } -jump-prev-custom-icon `
}
2018-03-22 09:23:51 +00:00
jumpPrev = (
< li
title = { this . showTitle ? prevItemTitle : null }
key = 'prev'
onClick = { this . jumpPrev }
tabIndex = '0'
onKeypress = { this . runIfEnterJumpPrev }
2018-11-08 11:59:09 +00:00
class = { jumpPrevClassString }
2018-03-22 09:23:51 +00:00
>
{ this . itemRender (
2018-11-08 11:59:09 +00:00
this . getJumpPrevPage ( ) ,
'jump-prev' ,
this . getItemIcon ( 'jumpPrevIcon' )
2018-03-22 09:23:51 +00:00
) }
< / li >
)
2018-11-08 11:59:09 +00:00
let jumpNextClassString = ` ${ prefixCls } -jump-next `
if ( props . jumpNextIcon ) {
jumpNextClassString += ` ${ prefixCls } -jump-next-custom-icon `
}
2018-03-22 09:23:51 +00:00
jumpNext = (
< li
title = { this . showTitle ? nextItemTitle : null }
key = 'next'
tabIndex = '0'
onClick = { this . jumpNext }
onKeypress = { this . runIfEnterJumpNext }
2018-11-08 11:59:09 +00:00
class = { jumpNextClassString }
2018-03-22 09:23:51 +00:00
>
{ this . itemRender (
2018-11-08 11:59:09 +00:00
this . getJumpNextPage ( ) ,
'jump-next' ,
this . getItemIcon ( 'jumpNextIcon' )
2018-03-22 09:23:51 +00:00
) }
< / li >
)
}
2018-03-05 11:06:44 +00:00
lastPager = (
< Pager
locale = { locale }
last
rootPrefixCls = { prefixCls }
onClick = { this . handleChange }
onKeypress = { this . runIfEnter }
key = { allPages }
page = { allPages }
active = { false }
showTitle = { this . showTitle }
itemRender = { this . itemRender }
/ >
)
firstPager = (
< Pager
locale = { locale }
rootPrefixCls = { prefixCls }
onClick = { this . handleChange }
onKeypress = { this . runIfEnter }
key = { 1 }
page = { 1 }
active = { false }
showTitle = { this . showTitle }
itemRender = { this . itemRender }
/ >
)
let left = Math . max ( 1 , stateCurrent - pageBufferSize )
let right = Math . min ( stateCurrent + pageBufferSize , allPages )
if ( stateCurrent - 1 <= pageBufferSize ) {
right = 1 + pageBufferSize * 2
}
if ( allPages - stateCurrent <= pageBufferSize ) {
left = allPages - pageBufferSize * 2
}
for ( let i = left ; i <= right ; i ++ ) {
const active = stateCurrent === i
pagerList . push (
< Pager
locale = { locale }
rootPrefixCls = { prefixCls }
onClick = { this . handleChange }
onKeypress = { this . runIfEnter }
key = { i }
page = { i }
active = { active }
showTitle = { this . showTitle }
itemRender = { this . itemRender }
/ >
)
}
if ( stateCurrent - 1 >= pageBufferSize * 2 && stateCurrent !== 1 + 2 ) {
pagerList [ 0 ] = (
< Pager
locale = { locale }
rootPrefixCls = { prefixCls }
onClick = { this . handleChange }
onKeypress = { this . runIfEnter }
key = { left }
page = { left }
class = { ` ${ prefixCls } -item-after-jump-prev ` }
active = { false }
showTitle = { this . showTitle }
itemRender = { this . itemRender }
/ >
)
pagerList . unshift ( jumpPrev )
}
if ( allPages - stateCurrent >= pageBufferSize * 2 && stateCurrent !== allPages - 2 ) {
pagerList [ pagerList . length - 1 ] = (
< Pager
locale = { locale }
rootPrefixCls = { prefixCls }
onClick = { this . handleChange }
onKeypress = { this . runIfEnter }
key = { right }
page = { right }
class = { ` ${ prefixCls } -item-before-jump-next ` }
active = { false }
showTitle = { this . showTitle }
itemRender = { this . itemRender }
/ >
)
pagerList . push ( jumpNext )
}
if ( left !== 1 ) {
pagerList . unshift ( firstPager )
}
if ( right !== allPages ) {
pagerList . push ( lastPager )
}
}
let totalText = null
if ( this . showTotal ) {
totalText = (
< li class = { ` ${ prefixCls } -total-text ` } >
{ this . showTotal (
this . total ,
[
( stateCurrent - 1 ) * statePageSize + 1 ,
stateCurrent * statePageSize > this . total ? this . total : stateCurrent * statePageSize ,
]
) }
< / li >
)
}
2019-01-01 13:19:50 +00:00
const prevDisabled = ! this . hasPrev ( ) || ! allPages
const nextDisabled = ! this . hasNext ( ) || ! allPages
2018-03-06 07:40:42 +00:00
const buildOptionText = this . buildOptionText || this . $scopedSlots . buildOptionText
2018-03-05 11:06:44 +00:00
return (
< ul
class = { ` ${ prefixCls } ` }
unselectable = 'unselectable'
2018-04-07 10:52:02 +00:00
ref = 'paginationNode'
2018-03-05 11:06:44 +00:00
>
{ totalText }
< li
title = { this . showTitle ? locale . prev _page : null }
onClick = { this . prev }
2018-03-09 04:20:21 +00:00
tabIndex = { prevDisabled ? null : 0 }
2018-03-05 11:06:44 +00:00
onKeypress = { this . runIfEnterPrev }
class = { ` ${ ! prevDisabled ? '' : ` ${ prefixCls } -disabled ` } ${ prefixCls } -prev ` }
aria - disabled = { prevDisabled }
>
2018-11-08 11:59:09 +00:00
{ this . itemRender ( prevPage ,
'prev' ,
this . getItemIcon ( 'prevIcon' )
) }
2018-03-05 11:06:44 +00:00
< / li >
{ pagerList }
< li
title = { this . showTitle ? locale . next _page : null }
onClick = { this . next }
2018-03-09 04:20:21 +00:00
tabIndex = { nextDisabled ? null : 0 }
2018-03-05 11:06:44 +00:00
onKeypress = { this . runIfEnterNext }
class = { ` ${ ! nextDisabled ? '' : ` ${ prefixCls } -disabled ` } ${ prefixCls } -next ` }
aria - disabled = { nextDisabled }
>
2018-11-08 11:59:09 +00:00
{ this . itemRender ( nextPage ,
'next' ,
this . getItemIcon ( 'nextIcon' )
) }
2018-03-05 11:06:44 +00:00
< / li >
< Options
locale = { locale }
rootPrefixCls = { prefixCls }
selectComponentClass = { this . selectComponentClass }
selectPrefixCls = { this . selectPrefixCls }
changeSize = { this . showSizeChanger ? this . changePageSize : null }
current = { stateCurrent }
pageSize = { statePageSize }
pageSizeOptions = { this . pageSizeOptions }
2018-03-06 07:40:42 +00:00
buildOptionText = { buildOptionText || null }
2018-03-05 11:06:44 +00:00
quickGo = { this . showQuickJumper ? this . handleChange : null }
goButton = { goButton }
/ >
< / ul >
)
} ,
}
2018-03-19 02:16:27 +00:00