bootstrap-table
Version:
An extended table to integration with some of the most widely used CSS frameworks. (Supports Bootstrap, Semantic UI, Bulma, Material Design, Foundation)
1,465 lines (1,252 loc) • 88.4 kB
JavaScript
/**
* @author zhixin wen <wenzhixin2010@gmail.com>
* version: 1.15.2
* https://github.com/wenzhixin/bootstrap-table/
*/
import Constants from './constants/index.js'
import Utils from './utils/index.js'
import VirtualScroll from './virtual-scroll/index.js'
class BootstrapTable {
constructor (el, options) {
this.options = options
this.$el = $(el)
this.$el_ = this.$el.clone()
this.timeoutId_ = 0
this.timeoutFooter_ = 0
this.init()
}
init () {
this.initConstants()
this.initLocale()
this.initContainer()
this.initTable()
this.initHeader()
this.initData()
this.initHiddenRows()
this.initToolbar()
this.initPagination()
this.initBody()
this.initSearchText()
this.initServer()
}
initConstants () {
const o = this.options
this.constants = Constants.CONSTANTS
this.constants.theme = $.fn.bootstrapTable.theme
const buttonsPrefix = o.buttonsPrefix ? `${o.buttonsPrefix}-` : ''
this.constants.buttonsClass = [
o.buttonsPrefix,
buttonsPrefix + o.buttonsClass,
Utils.sprintf(`${buttonsPrefix}%s`, o.iconSize)
].join(' ').trim()
}
initLocale () {
if (this.options.locale) {
const locales = $.fn.bootstrapTable.locales
const parts = this.options.locale.split(/-|_/)
parts[0] = parts[0].toLowerCase()
if (parts[1]) {
parts[1] = parts[1].toUpperCase()
}
if (locales[this.options.locale]) {
$.extend(this.options, locales[this.options.locale])
} else if (locales[parts.join('-')]) {
$.extend(this.options, locales[parts.join('-')])
} else if (locales[parts[0]]) {
$.extend(this.options, locales[parts[0]])
}
}
}
initContainer () {
const topPagination = ['top', 'both'].includes(this.options.paginationVAlign)
? '<div class="fixed-table-pagination clearfix"></div>' : ''
const bottomPagination = ['bottom', 'both'].includes(this.options.paginationVAlign)
? '<div class="fixed-table-pagination"></div>' : ''
this.$container = $(`
<div class="bootstrap-table ${this.constants.theme}">
<div class="fixed-table-toolbar"></div>
${topPagination}
<div class="fixed-table-container">
<div class="fixed-table-header"><table></table></div>
<div class="fixed-table-body">
<div class="fixed-table-loading">
<span class="loading-wrap">
<span class="loading-text">${this.options.formatLoadingMessage()}</span>
<span class="animation-wrap"><span class="animation-dot"></span></span>
</span>
</div>
</div>
<div class="fixed-table-footer"><table><thead><tr></tr></thead></table></div>
</div>
${bottomPagination}
</div>
`)
this.$container.insertAfter(this.$el)
this.$tableContainer = this.$container.find('.fixed-table-container')
this.$tableHeader = this.$container.find('.fixed-table-header')
this.$tableBody = this.$container.find('.fixed-table-body')
this.$tableLoading = this.$container.find('.fixed-table-loading')
this.$tableFooter = this.$el.find('tfoot')
// checking if custom table-toolbar exists or not
if (this.options.buttonsToolbar) {
this.$toolbar = $('body').find(this.options.buttonsToolbar)
} else {
this.$toolbar = this.$container.find('.fixed-table-toolbar')
}
this.$pagination = this.$container.find('.fixed-table-pagination')
this.$tableBody.append(this.$el)
this.$container.after('<div class="clearfix"></div>')
this.$el.addClass(this.options.classes)
this.$tableLoading.addClass(this.options.classes)
if (this.options.height) {
this.$tableContainer.addClass('fixed-height')
if (this.options.showFooter) {
this.$tableContainer.addClass('has-footer')
}
if (this.options.classes.split(' ').includes('table-bordered')) {
this.$tableBody.append('<div class="fixed-table-border"></div>')
this.$tableBorder = this.$tableBody.find('.fixed-table-border')
this.$tableLoading.addClass('fixed-table-border')
}
this.$tableFooter = this.$container.find('.fixed-table-footer')
}
}
initTable () {
const columns = []
const data = []
this.$header = this.$el.find('>thead')
if (!this.$header.length) {
this.$header = $(`<thead class="${this.options.theadClasses}"></thead>`).appendTo(this.$el)
} else if (this.options.theadClasses) {
this.$header.addClass(this.options.theadClasses)
}
this.$header.find('tr').each((i, el) => {
const column = []
$(el).find('th').each((i, el) => {
// #2014: getFieldIndex and elsewhere assume this is string, causes issues if not
if (typeof $(el).data('field') !== 'undefined') {
$(el).data('field', `${$(el).data('field')}`)
}
column.push($.extend({}, {
title: $(el).html(),
'class': $(el).attr('class'),
titleTooltip: $(el).attr('title'),
rowspan: $(el).attr('rowspan') ? +$(el).attr('rowspan') : undefined,
colspan: $(el).attr('colspan') ? +$(el).attr('colspan') : undefined
}, $(el).data()))
})
columns.push(column)
})
if (!Array.isArray(this.options.columns[0])) {
this.options.columns = [this.options.columns]
}
this.options.columns = $.extend(true, [], columns, this.options.columns)
this.columns = []
this.fieldsColumnsIndex = []
Utils.setFieldIndex(this.options.columns)
this.options.columns.forEach((columns, i) => {
columns.forEach((_column, j) => {
const column = $.extend({}, BootstrapTable.COLUMN_DEFAULTS, _column)
if (typeof column.fieldIndex !== 'undefined') {
this.columns[column.fieldIndex] = column
this.fieldsColumnsIndex[column.field] = column.fieldIndex
}
this.options.columns[i][j] = column
})
})
// if options.data is setting, do not process tbody and tfoot data
if (!this.options.data.length) {
this.options.data = Utils.trToData(this.columns, this.$el.find('>tbody>tr'))
if (data.length) {
this.fromHtml = true
}
}
this.footerData = Utils.trToData(this.columns, this.$el.find('>tfoot>tr'))
if (this.footerData) {
this.$el.find('tfoot').html('<tr></tr>')
}
if (!this.options.showFooter || this.options.cardView) {
this.$tableFooter.hide()
} else {
this.$tableFooter.show()
}
}
initHeader () {
const visibleColumns = {}
const html = []
this.header = {
fields: [],
styles: [],
classes: [],
formatters: [],
detailFormatters: [],
events: [],
sorters: [],
sortNames: [],
cellStyles: [],
searchables: []
}
this.options.columns.forEach((columns, i) => {
html.push('<tr>')
if (i === 0 && !this.options.cardView && this.options.detailView && this.options.detailViewIcon) {
html.push(`<th class="detail" rowspan="${this.options.columns.length}">
<div class="fht-cell"></div>
</th>
`)
}
columns.forEach((column, j) => {
const class_ = Utils.sprintf(' class="%s"', column['class'])
const unitWidth = column.widthUnit
const width = Number.parseFloat(column.width)
const halign = Utils.sprintf('text-align: %s; ', column.halign ? column.halign : column.align)
const align = Utils.sprintf('text-align: %s; ', column.align)
let style = Utils.sprintf('vertical-align: %s; ', column.valign)
style += Utils.sprintf('width: %s; ', (column.checkbox || column.radio) && !width
? (!column.showSelectTitle ? '36px' : undefined)
: (width ? width + unitWidth : undefined))
if (typeof column.fieldIndex !== 'undefined') {
this.header.fields[column.fieldIndex] = column.field
this.header.styles[column.fieldIndex] = align + style
this.header.classes[column.fieldIndex] = class_
this.header.formatters[column.fieldIndex] = column.formatter
this.header.detailFormatters[column.fieldIndex] = column.detailFormatter
this.header.events[column.fieldIndex] = column.events
this.header.sorters[column.fieldIndex] = column.sorter
this.header.sortNames[column.fieldIndex] = column.sortName
this.header.cellStyles[column.fieldIndex] = column.cellStyle
this.header.searchables[column.fieldIndex] = column.searchable
if (!column.visible) {
return
}
if (this.options.cardView && (!column.cardVisible)) {
return
}
visibleColumns[column.field] = column
}
html.push(`<th${Utils.sprintf(' title="%s"', column.titleTooltip)}`,
column.checkbox || column.radio
? Utils.sprintf(' class="bs-checkbox %s"', column['class'] || '')
: class_,
Utils.sprintf(' style="%s"', halign + style),
Utils.sprintf(' rowspan="%s"', column.rowspan),
Utils.sprintf(' colspan="%s"', column.colspan),
Utils.sprintf(' data-field="%s"', column.field),
// If `column` is not the first element of `this.options.columns[0]`, then className 'data-not-first-th' should be added.
j === 0 && i > 0 ? ' data-not-first-th' : '',
'>')
html.push(Utils.sprintf('<div class="th-inner %s">', this.options.sortable && column.sortable
? 'sortable both' : ''))
let text = this.options.escape ? Utils.escapeHTML(column.title) : column.title
const title = text
if (column.checkbox) {
text = ''
if (!this.options.singleSelect && this.options.checkboxHeader) {
text = '<label><input name="btSelectAll" type="checkbox" /><span></span></label>'
}
this.header.stateField = column.field
}
if (column.radio) {
text = ''
this.header.stateField = column.field
this.options.singleSelect = true
}
if (!text && column.showSelectTitle) {
text += title
}
html.push(text)
html.push('</div>')
html.push('<div class="fht-cell"></div>')
html.push('</div>')
html.push('</th>')
})
html.push('</tr>')
})
this.$header.html(html.join(''))
this.$header.find('th[data-field]').each((i, el) => {
$(el).data(visibleColumns[$(el).data('field')])
})
this.$container.off('click', '.th-inner').on('click', '.th-inner', e => {
const $this = $(e.currentTarget)
if (this.options.detailView && !$this.parent().hasClass('bs-checkbox')) {
if ($this.closest('.bootstrap-table')[0] !== this.$container[0]) {
return false
}
}
if (this.options.sortable && $this.parent().data().sortable) {
this.onSort(e)
}
})
this.$header.children().children().off('keypress').on('keypress', e => {
if (this.options.sortable && $(e.currentTarget).data().sortable) {
const code = e.keyCode || e.which
if (code === 13) { // Enter keycode
this.onSort(e)
}
}
})
const resizeEvent = `resize.bootstrap-table${this.$el.attr('id') || ''}`
$(window).off(resizeEvent)
if (!this.options.showHeader || this.options.cardView) {
this.$header.hide()
this.$tableHeader.hide()
this.$tableLoading.css('top', 0)
} else {
this.$header.show()
this.$tableHeader.show()
this.$tableLoading.css('top', this.$header.outerHeight() + 1)
// Assign the correct sortable arrow
this.getCaret()
$(window).on(resizeEvent, e => this.resetWidth(e))
}
this.$selectAll = this.$header.find('[name="btSelectAll"]')
this.$selectAll.off('click').on('click', ({currentTarget}) => {
const checked = $(currentTarget).prop('checked')
this[checked ? 'checkAll' : 'uncheckAll']()
this.updateSelected()
})
}
initData (data, type) {
if (type === 'append') {
this.options.data = this.options.data.concat(data)
} else if (type === 'prepend') {
this.options.data = [].concat(data).concat(this.options.data)
} else {
this.options.data = data || this.options.data
}
this.data = this.options.data
if (this.options.sidePagination === 'server') {
return
}
this.initSort()
}
initSort () {
let name = this.options.sortName
const order = this.options.sortOrder === 'desc' ? -1 : 1
const index = this.header.fields.indexOf(this.options.sortName)
let timeoutId = 0
if (index !== -1) {
if (this.options.sortStable) {
this.data.forEach((row, i) => {
if (!row.hasOwnProperty('_position')) {
row._position = i
}
})
}
if (this.options.customSort) {
Utils.calculateObjectValue(this.options, this.options.customSort, [
this.options.sortName,
this.options.sortOrder,
this.data
])
} else {
this.data.sort((a, b) => {
if (this.header.sortNames[index]) {
name = this.header.sortNames[index]
}
const aa = Utils.getItemField(a, name, this.options.escape)
const bb = Utils.getItemField(b, name, this.options.escape)
const value = Utils.calculateObjectValue(this.header, this.header.sorters[index], [aa, bb, a, b])
if (value !== undefined) {
if (this.options.sortStable && value === 0) {
return order * (a._position - b._position)
}
return order * value
}
return Utils.sort(aa, bb, order, this.options.sortStable)
})
}
if (this.options.sortClass !== undefined) {
clearTimeout(timeoutId)
timeoutId = setTimeout(() => {
this.$el.removeClass(this.options.sortClass)
const index = this.$header.find(`[data-field="${this.options.sortName}"]`).index()
this.$el.find(`tr td:nth-child(${index + 1})`).addClass(this.options.sortClass)
}, 250)
}
}
}
onSort ({type, currentTarget}) {
const $this = type === 'keypress' ? $(currentTarget) : $(currentTarget).parent()
const $this_ = this.$header.find('th').eq($this.index())
this.$header.add(this.$header_).find('span.order').remove()
if (this.options.sortName === $this.data('field')) {
this.options.sortOrder = this.options.sortOrder === 'asc' ? 'desc' : 'asc'
} else {
this.options.sortName = $this.data('field')
if (this.options.rememberOrder) {
this.options.sortOrder = $this.data('order') === 'asc' ? 'desc' : 'asc'
} else {
this.options.sortOrder = this.columns[this.fieldsColumnsIndex[$this.data('field')]].sortOrder ||
this.columns[this.fieldsColumnsIndex[$this.data('field')]].order
}
}
this.trigger('sort', this.options.sortName, this.options.sortOrder)
$this.add($this_).data('order', this.options.sortOrder)
// Assign the correct sortable arrow
this.getCaret()
if (this.options.sidePagination === 'server') {
this.options.pageNumber = 1
this.initServer(this.options.silentSort)
return
}
this.initSort()
this.initBody()
}
initToolbar () {
const o = this.options
let html = []
let timeoutId = 0
let $keepOpen
let $search
let switchableCount = 0
if (this.$toolbar.find('.bs-bars').children().length) {
$('body').append($(o.toolbar))
}
this.$toolbar.html('')
if (typeof o.toolbar === 'string' || typeof o.toolbar === 'object') {
$(Utils.sprintf('<div class="bs-bars %s-%s"></div>', this.constants.classes.pull, o.toolbarAlign))
.appendTo(this.$toolbar)
.append($(o.toolbar))
}
// showColumns, showToggle, showRefresh
html = [`<div class="${[
'columns',
`columns-${o.buttonsAlign}`,
this.constants.classes.buttonsGroup,
`${this.constants.classes.pull}-${o.buttonsAlign}`
].join(' ')}">`]
if (typeof o.icons === 'string') {
o.icons = Utils.calculateObjectValue(null, o.icons)
}
if (o.showPaginationSwitch) {
html.push(`<button class="${this.constants.buttonsClass}" type="button" name="paginationSwitch"
aria-label="Pagination Switch" title="${o.formatPaginationSwitch()}">
${o.showButtonIcons ? Utils.sprintf(this.constants.html.icon, o.iconsPrefix, o.icons.paginationSwitchDown) : ''}
${o.showButtonText ? o.formatPaginationSwitchUp() : ''}
</button>`)
}
if (o.showRefresh) {
html.push(`<button class="${this.constants.buttonsClass}" type="button" name="refresh"
aria-label="Refresh" title="${o.formatRefresh()}">
${o.showButtonIcons ? Utils.sprintf(this.constants.html.icon, o.iconsPrefix, o.icons.refresh) : ''}
${o.showButtonText ? o.formatRefresh() : ''}
</button>`)
}
if (o.showToggle) {
html.push(`<button class="${this.constants.buttonsClass}" type="button" name="toggle"
aria-label="Toggle" title="${o.formatToggle()}">
${o.showButtonIcons ? Utils.sprintf(this.constants.html.icon, o.iconsPrefix, o.icons.toggleOff) : '' }
${o.showButtonText ? o.formatToggleOn() : ''}
</button>`)
}
if (o.showFullscreen) {
html.push(`<button class="${this.constants.buttonsClass}" type="button" name="fullscreen"
aria-label="Fullscreen" title="${o.formatFullscreen()}">
${o.showButtonIcons ? Utils.sprintf(this.constants.html.icon, o.iconsPrefix, o.icons.fullscreen) : '' }
${o.showButtonText ? o.formatFullscreen() : ''}
</button>`)
}
if (o.showColumns) {
html.push(`<div class="keep-open ${this.constants.classes.buttonsDropdown}" title="${o.formatColumns()}">
<button class="${this.constants.buttonsClass} dropdown-toggle" type="button" data-toggle="dropdown"
aria-label="Columns" title="${o.formatColumns()}">
${o.showButtonIcons ? Utils.sprintf(this.constants.html.icon, o.iconsPrefix, o.icons.columns) : '' }
${o.showButtonText ? o.formatColumns() : ''}
${this.constants.html.dropdownCaret}
</button>
${this.constants.html.toolbarDropdown[0]}`)
if (o.showColumnsToggleAll) {
const allFieldsVisible = this.getVisibleColumns().length === this.columns.length
html.push(
Utils.sprintf(this.constants.html.toolbarDropdownItem,
Utils.sprintf('<input type="checkbox" class="toggle-all" %s> <span>%s</span>', allFieldsVisible ? 'checked="checked"' : '', o.formatColumnsToggleAll())
)
)
html.push(this.constants.html.toolbarDropdownSeperator)
}
this.columns.forEach((column, i) => {
if (column.radio || column.checkbox) {
return
}
if (o.cardView && !column.cardVisible) {
return
}
const checked = column.visible ? ' checked="checked"' : ''
if (column.switchable) {
html.push(Utils.sprintf(this.constants.html.toolbarDropdownItem,
Utils.sprintf('<input type="checkbox" data-field="%s" value="%s"%s> <span>%s</span>',
column.field, i, checked, column.title)))
switchableCount++
}
})
html.push(this.constants.html.toolbarDropdown[1], '</div>')
}
html.push('</div>')
// Fix #188: this.showToolbar is for extensions
if (this.showToolbar || html.length > 2) {
this.$toolbar.append(html.join(''))
}
if (o.showPaginationSwitch) {
this.$toolbar.find('button[name="paginationSwitch"]')
.off('click').on('click', () => this.togglePagination())
}
if (o.showFullscreen) {
this.$toolbar.find('button[name="fullscreen"]')
.off('click').on('click', () => this.toggleFullscreen())
}
if (o.showRefresh) {
this.$toolbar.find('button[name="refresh"]')
.off('click').on('click', () => this.refresh())
}
if (o.showToggle) {
this.$toolbar.find('button[name="toggle"]')
.off('click').on('click', () => {
this.toggleView()
})
}
if (o.showColumns) {
$keepOpen = this.$toolbar.find('.keep-open')
const $checkboxes = $keepOpen.find('input:not(".toggle-all")')
const $toggleAll = $keepOpen.find('input.toggle-all')
if (switchableCount <= o.minimumCountColumns) {
$keepOpen.find('input').prop('disabled', true)
}
$keepOpen.find('li, label').off('click').on('click', e => {
e.stopImmediatePropagation()
})
$checkboxes.off('click').on('click', ({currentTarget}) => {
const $this = $(currentTarget)
this._toggleColumn($this.val(), $this.prop('checked'), false)
this.trigger('column-switch', $this.data('field'), $this.prop('checked'))
$toggleAll.prop('checked', $checkboxes.filter(':checked').length === this.columns.length)
})
$toggleAll.off('click').on('click', ({currentTarget}) => {
this._toggleAllColumns($(currentTarget).prop('checked'))
})
}
if (o.search) {
html = []
const showSearchButton = Utils.sprintf(this.constants.html.searchButton, o.formatSearch(), o.showButtonIcons ? Utils.sprintf(this.constants.html.icon, o.iconsPrefix, o.icons.search) : '', o.showButtonText ? o.formatSearch() : '')
const showSearchClearButton = Utils.sprintf(this.constants.html.searchClearButton, o.formatClearSearch(), o.showButtonIcons ? Utils.sprintf(this.constants.html.icon, o.iconsPrefix, o.icons.clearSearch) : '', o.showButtonText ? o.formatClearSearch() : '')
const searchInputHtml = `<input class="${this.constants.classes.input}${Utils.sprintf(' input-%s', o.iconSize)} search-input" type="text" placeholder="${o.formatSearch()}">`
let searchInputFinalHtml = searchInputHtml
if (o.showSearchButton || o.showSearchClearButton) {
searchInputFinalHtml = Utils.sprintf(this.constants.html.inputGroup,
searchInputHtml,
(o.showSearchButton ? showSearchButton : '') +
(o.showSearchClearButton ? showSearchClearButton : ''))
}
html.push(Utils.sprintf(`
<div class="${this.constants.classes.pull}-${o.searchAlign} search ${this.constants.classes.inputGroup}">
%s
</div>
`, searchInputFinalHtml))
this.$toolbar.append(html.join(''))
const $searchInput = this.$toolbar.find('.search input')
$search = o.showSearchButton ? this.$toolbar.find('.search button[name=search]') : $searchInput
const eventTriggers = o.showSearchButton ? 'click' :
(Utils.isIEBrowser() ? 'mouseup' : 'keyup drop blur')
$search.off(eventTriggers).on(eventTriggers, event => {
if (o.searchOnEnterKey && event.keyCode !== 13) {
return
}
if ([37, 38, 39, 40].includes(event.keyCode)) {
return
}
clearTimeout(timeoutId) // doesn't matter if it's 0
timeoutId = setTimeout(() => {
this.onSearch(o.showSearchButton ? {currentTarget: $searchInput} : event)
}, o.searchTimeOut)
})
if (o.showSearchClearButton) {
this.$toolbar.find('.search button[name=clearSearch]').click(() => {
this.resetSearch()
this.onSearch({currentTarget: this.$toolbar.find('.search input')})
})
}
}
}
onSearch ({currentTarget, firedByInitSearchText} = {}, overwriteSearchText = true) {
if (currentTarget !== undefined && overwriteSearchText) {
const text = $(currentTarget).val().trim()
if (this.options.trimOnSearch && $(currentTarget).val() !== text) {
$(currentTarget).val(text)
}
if (this.searchText === text) {
return
}
if ($(currentTarget).hasClass('search-input')) {
this.searchText = text
this.options.searchText = text
}
}
if (!firedByInitSearchText) {
this.options.pageNumber = 1
}
this.initSearch()
if (firedByInitSearchText) {
if (this.options.sidePagination === 'client') {
this.updatePagination()
}
} else {
this.updatePagination()
}
this.trigger('search', this.searchText)
}
initSearch () {
this.filterOptions = this.filterOptions || this.options.filterOptions
if (this.options.sidePagination !== 'server') {
if (this.options.customSearch) {
this.data = Utils.calculateObjectValue(this.options, this.options.customSearch,
[this.options.data, this.searchText])
return
}
const s = this.searchText && (this.options.escape
? Utils.escapeHTML(this.searchText) : this.searchText).toLowerCase()
const f = Utils.isEmptyObject(this.filterColumns) ? null : this.filterColumns
// Check filter
if (typeof this.filterOptions.filterAlgorithm === 'function') {
this.data = this.options.data.filter((item, i) => this.filterOptions.filterAlgorithm.apply(null, [item, f]))
} else if (typeof this.filterOptions.filterAlgorithm === 'string') {
this.data = f ? this.options.data.filter((item, i) => {
const filterAlgorithm = this.filterOptions.filterAlgorithm
if (filterAlgorithm === 'and') {
for (const key in f) {
if (
(Array.isArray(f[key]) &&
!f[key].includes(item[key])) ||
(!Array.isArray(f[key]) &&
item[key] !== f[key])
) {
return false
}
}
} else if (filterAlgorithm === 'or') {
let match = false
for (const key in f) {
if (
(Array.isArray(f[key]) &&
f[key].includes(item[key])) ||
(!Array.isArray(f[key]) &&
item[key] === f[key])
) {
match = true
}
}
return match
}
return true
}) : this.options.data
}
const visibleFields = this.getVisibleFields()
this.data = s ? this.data.filter((item, i) => {
for (let j = 0; j < this.header.fields.length; j++) {
if (!this.header.searchables[j] || (this.options.visibleSearch && visibleFields.indexOf(this.header.fields[j]) === -1)) {
continue
}
const key = Utils.isNumeric(this.header.fields[j]) ? parseInt(this.header.fields[j], 10) : this.header.fields[j]
const column = this.columns[this.fieldsColumnsIndex[key]]
let value
if (typeof key === 'string') {
value = item
const props = key.split('.')
for (let i = 0; i < props.length; i++) {
if (value[props[i]] !== null) {
value = value[props[i]]
}
}
} else {
value = item[key]
}
// Fix #142: respect searchFormatter boolean
if (column && column.searchFormatter) {
value = Utils.calculateObjectValue(column,
this.header.formatters[j], [value, item, i, column.field], value)
}
if (typeof value === 'string' || typeof value === 'number') {
if (this.options.strictSearch) {
if ((`${value}`).toLowerCase() === s) {
return true
}
} else {
const largerSmallerEqualsRegex = /(?:(<=|=>|=<|>=|>|<)(?:\s+)?(\d+)?|(\d+)?(\s+)?(<=|=>|=<|>=|>|<))/gm
const matches = largerSmallerEqualsRegex.exec(s)
let comparisonCheck = false
if (matches) {
const operator = matches[1] || `${matches[5]}l`
const comparisonValue = matches[2] || matches[3]
const int = parseInt(value, 10)
const comparisonInt = parseInt(comparisonValue, 10)
switch (operator) {
case '>':
case '<l':
comparisonCheck = int > comparisonInt
break
case '<':
case '>l':
comparisonCheck = int < comparisonInt
break
case '<=':
case '=<':
case '>=l':
case '=>l':
comparisonCheck = int <= comparisonInt
break
case '>=':
case '=>':
case '<=l':
case '=<l':
comparisonCheck = int >= comparisonInt
break
default:
break
}
}
if (comparisonCheck || (`${value}`).toLowerCase().includes(s)) {
return true
}
}
}
}
return false
}) : this.data
}
}
initPagination () {
const o = this.options
if (!o.pagination) {
this.$pagination.hide()
return
}
this.$pagination.show()
const html = []
let $allSelected = false
let i
let from
let to
let $pageList
let $pre
let $next
let $number
const data = this.getData({includeHiddenRows: false})
let pageList = o.pageList
if (o.sidePagination !== 'server') {
o.totalRows = data.length
}
this.totalPages = 0
if (o.totalRows) {
if (o.pageSize === o.formatAllRows()) {
o.pageSize = o.totalRows
$allSelected = true
} else if (o.pageSize === o.totalRows) {
// Fix #667 Table with pagination,
// multiple pages and a search this matches to one page throws exception
const pageLst = typeof o.pageList === 'string'
? o.pageList.replace('[', '').replace(']', '')
.replace(/ /g, '').toLowerCase().split(',') : o.pageList
if (pageLst.includes(o.formatAllRows().toLowerCase())) {
$allSelected = true
}
}
this.totalPages = ~~((o.totalRows - 1) / o.pageSize) + 1
o.totalPages = this.totalPages
}
if (this.totalPages > 0 && o.pageNumber > this.totalPages) {
o.pageNumber = this.totalPages
}
this.pageFrom = (o.pageNumber - 1) * o.pageSize + 1
this.pageTo = o.pageNumber * o.pageSize
if (this.pageTo > o.totalRows) {
this.pageTo = o.totalRows
}
if (this.options.pagination && this.options.sidePagination !== 'server') {
this.options.totalNotFiltered = this.options.data.length
}
if (!this.options.showExtendedPagination) {
this.options.totalNotFiltered = undefined
}
const paginationInfo = o.onlyInfoPagination ?
o.formatDetailPagination(o.totalRows) :
o.formatShowingRows(this.pageFrom, this.pageTo, o.totalRows, o.totalNotFiltered)
html.push(`<div class="${this.constants.classes.pull}-${o.paginationDetailHAlign} pagination-detail">
<span class="pagination-info">
${paginationInfo}
</span>`)
if (!o.onlyInfoPagination) {
html.push('<span class="page-list">')
const pageNumber = [
`<span class="${this.constants.classes.paginationDropdown}">
<button class="${this.constants.buttonsClass} dropdown-toggle" type="button" data-toggle="dropdown">
<span class="page-size">
${$allSelected ? o.formatAllRows() : o.pageSize}
</span>
${this.constants.html.dropdownCaret}
</button>
${this.constants.html.pageDropdown[0]}`]
if (typeof o.pageList === 'string') {
const list = o.pageList.replace('[', '').replace(']', '')
.replace(/ /g, '').split(',')
pageList = []
for (const value of list) {
pageList.push(
(value.toLowerCase() === o.formatAllRows().toLowerCase() ||
['all', 'unlimited'].includes(value.toLowerCase()))
? o.formatAllRows() : +value)
}
}
pageList.forEach((page, i) => {
if (!o.smartDisplay || i === 0 || pageList[i - 1] < o.totalRows) {
let active
if ($allSelected) {
active = page === o.formatAllRows() ? this.constants.classes.dropdownActive : ''
} else {
active = page === o.pageSize ? this.constants.classes.dropdownActive : ''
}
pageNumber.push(Utils.sprintf(this.constants.html.pageDropdownItem, active, page))
}
})
pageNumber.push(`${this.constants.html.pageDropdown[1]}</span>`)
html.push(o.formatRecordsPerPage(pageNumber.join('')))
html.push('</span></div>')
html.push(`<div class="${this.constants.classes.pull}-${o.paginationHAlign} pagination">`,
Utils.sprintf(this.constants.html.pagination[0], Utils.sprintf(' pagination-%s', o.iconSize)),
Utils.sprintf(this.constants.html.paginationItem, ' page-pre', o.formatSRPaginationPreText(), o.paginationPreText))
if (this.totalPages < o.paginationSuccessivelySize) {
from = 1
to = this.totalPages
} else {
from = o.pageNumber - o.paginationPagesBySide
to = from + (o.paginationPagesBySide * 2)
}
if (o.pageNumber < (o.paginationSuccessivelySize - 1)) {
to = o.paginationSuccessivelySize
}
if (o.paginationSuccessivelySize > this.totalPages - from) {
from = from - (o.paginationSuccessivelySize - (this.totalPages - from)) + 1
}
if (from < 1) {
from = 1
}
if (to > this.totalPages) {
to = this.totalPages
}
const middleSize = Math.round(o.paginationPagesBySide / 2)
const pageItem = (i, classes = '') => Utils.sprintf(this.constants.html.paginationItem,
classes + (i === o.pageNumber ? ` ${this.constants.classes.paginationActive}` : ''), o.formatSRPaginationPageText(i), i)
if (from > 1) {
let max = o.paginationPagesBySide
if (max >= from) max = from - 1
for (i = 1; i <= max; i++) {
html.push(pageItem(i))
}
if ((from - 1) === max + 1) {
i = from - 1
html.push(pageItem(i))
} else {
if ((from - 1) > max) {
if (
(from - o.paginationPagesBySide * 2) > o.paginationPagesBySide &&
o.paginationUseIntermediate
) {
i = Math.round(((from - middleSize) / 2) + middleSize)
html.push(pageItem(i, ' page-intermediate'))
} else {
html.push(Utils.sprintf(this.constants.html.paginationItem,
' page-first-separator disabled', '', '...'))
}
}
}
}
for (i = from; i <= to; i++) {
html.push(pageItem(i))
}
if (this.totalPages > to) {
let min = this.totalPages - (o.paginationPagesBySide - 1)
if (to >= min) min = to + 1
if ((to + 1) === min - 1) {
i = to + 1
html.push(pageItem(i))
} else {
if (min > (to + 1)) {
if (
(this.totalPages - to) > o.paginationPagesBySide * 2 &&
o.paginationUseIntermediate
) {
i = Math.round(((this.totalPages - middleSize - to) / 2) + to)
html.push(pageItem(i, ' page-intermediate'))
} else {
html.push(Utils.sprintf(this.constants.html.paginationItem,
' page-last-separator disabled', '', '...'))
}
}
}
for (i = min; i <= this.totalPages; i++) {
html.push(pageItem(i))
}
}
html.push(Utils.sprintf(this.constants.html.paginationItem, ' page-next', o.formatSRPaginationNextText(), o.paginationNextText))
html.push(this.constants.html.pagination[1], '</div>')
}
this.$pagination.html(html.join(''))
const dropupClass = ['bottom', 'both'].includes(o.paginationVAlign) ?
` ${this.constants.classes.dropup}` : ''
this.$pagination.last().find('.page-list > span').addClass(dropupClass)
if (!o.onlyInfoPagination) {
$pageList = this.$pagination.find('.page-list a')
$pre = this.$pagination.find('.page-pre')
$next = this.$pagination.find('.page-next')
$number = this.$pagination.find('.page-item').not('.page-next, .page-pre, .page-last-separator, .page-first-separator')
if (this.totalPages <= 1) {
this.$pagination.find('div.pagination').hide()
}
if (o.smartDisplay) {
if (pageList.length < 2 || o.totalRows <= pageList[0]) {
this.$pagination.find('span.page-list').hide()
}
}
// when data is empty, hide the pagination
this.$pagination[this.getData().length ? 'show' : 'hide']()
if (!o.paginationLoop) {
if (o.pageNumber === 1) {
$pre.addClass('disabled')
}
if (o.pageNumber === this.totalPages) {
$next.addClass('disabled')
}
}
if ($allSelected) {
o.pageSize = o.formatAllRows()
}
// removed the events for last and first, onPageNumber executeds the same logic
$pageList.off('click').on('click', e => this.onPageListChange(e))
$pre.off('click').on('click', e => this.onPagePre(e))
$next.off('click').on('click', e => this.onPageNext(e))
$number.off('click').on('click', e => this.onPageNumber(e))
}
}
updatePagination (event) {
// Fix #171: IE disabled button can be clicked bug.
if (event && $(event.currentTarget).hasClass('disabled')) {
return
}
if (!this.options.maintainMetaData) {
this.resetRows()
}
this.initPagination()
if (this.options.sidePagination === 'server') {
this.initServer()
} else {
this.initBody()
}
this.trigger('page-change', this.options.pageNumber, this.options.pageSize)
}
onPageListChange (event) {
event.preventDefault()
const $this = $(event.currentTarget)
$this.parent().addClass(this.constants.classes.dropdownActive)
.siblings().removeClass(this.constants.classes.dropdownActive)
this.options.pageSize = $this.text().toUpperCase() === this.options.formatAllRows().toUpperCase()
? this.options.formatAllRows() : +$this.text()
this.$toolbar.find('.page-size').text(this.options.pageSize)
this.updatePagination(event)
return false
}
onPagePre (event) {
event.preventDefault()
if ((this.options.pageNumber - 1) === 0) {
this.options.pageNumber = this.options.totalPages
} else {
this.options.pageNumber--
}
this.updatePagination(event)
return false
}
onPageNext (event) {
event.preventDefault()
if ((this.options.pageNumber + 1) > this.options.totalPages) {
this.options.pageNumber = 1
} else {
this.options.pageNumber++
}
this.updatePagination(event)
return false
}
onPageNumber (event) {
event.preventDefault()
if (this.options.pageNumber === +$(event.currentTarget).text()) {
return
}
this.options.pageNumber = +$(event.currentTarget).text()
this.updatePagination(event)
return false
}
initRow (item, i, data, trFragments) {
const html = []
let style = {}
const csses = []
let data_ = ''
let attributes = {}
const htmlAttributes = []
if (Utils.findIndex(this.hiddenRows, item) > -1) {
return
}
style = Utils.calculateObjectValue(this.options, this.options.rowStyle, [item, i], style)
if (style && style.css) {
for (const [key, value] of Object.entries(style.css)) {
csses.push(`${key}: ${value}`)
}
}
attributes = Utils.calculateObjectValue(this.options,
this.options.rowAttributes, [item, i], attributes)
if (attributes) {
for (const [key, value] of Object.entries(attributes)) {
htmlAttributes.push(`${key}="${Utils.escapeHTML(value)}"`)
}
}
if (item._data && !Utils.isEmptyObject(item._data)) {
for (const [k, v] of Object.entries(item._data)) {
// ignore data-index
if (k === 'index') {
return
}
data_ += ` data-${k}='${typeof v === 'object' ? JSON.stringify(v) : v}'`
}
}
html.push('<tr',
Utils.sprintf(' %s', htmlAttributes.length ? htmlAttributes.join(' ') : undefined),
Utils.sprintf(' id="%s"', Array.isArray(item) ? undefined : item._id),
Utils.sprintf(' class="%s"', style.classes || (Array.isArray(item) ? undefined : item._class)),
` data-index="${i}"`,
Utils.sprintf(' data-uniqueid="%s"', Utils.getItemField(item, this.options.uniqueId, false)),
Utils.sprintf(' data-has-detail-view="%s"', (!this.options.cardView && this.options.detailView && Utils.calculateObjectValue(null, this.options.detailFilter, [i, item])) ? 'true' : undefined),
Utils.sprintf('%s', data_),
'>'
)
if (this.options.cardView) {
html.push(`<td colspan="${this.header.fields.length}"><div class="card-views">`)
}
if (!this.options.cardView && this.options.detailView && this.options.detailViewIcon) {
html.push('<td>')
if (Utils.calculateObjectValue(null, this.options.detailFilter, [i, item])) {
html.push(`
<a class="detail-icon" href="#">
${Utils.sprintf(this.constants.html.icon, this.options.iconsPrefix, this.options.icons.detailOpen)}
</a>
`)
}
html.push('</td>')
}
this.header.fields.forEach((field, j) => {
let text = ''
let value_ = Utils.getItemField(item, field, this.options.escape)
let value = ''
let type = ''
let cellStyle = {}
let id_ = ''
let class_ = this.header.classes[j]
let style_ = ''
let data_ = ''
let rowspan_ = ''
let colspan_ = ''
let title_ = ''
const column = this.columns[j]
if (this.fromHtml && typeof value_ === 'undefined') {
if ((!column.checkbox) && (!column.radio)) {
return
}
}
if (!column.visible) {
return
}
if (this.options.cardView && (!column.cardVisible)) {
return
}
if (column.escape) {
value_ = Utils.escapeHTML(value_)
}
if (csses.concat([this.header.styles[j]]).length) {
style_ = ` style="${csses.concat([this.header.styles[j]]).join('; ')}"`
}
// handle td's id and class
if (item[`_${field}_id`]) {
id_ = Utils.sprintf(' id="%s"', item[`_${field}_id`])
}
if (item[`_${field}_class`]) {
class_ = Utils.sprintf(' class="%s"', item[`_${field}_class`])
}
if (item[`_${field}_rowspan`]) {
rowspan_ = Utils.sprintf(' rowspan="%s"', item[`_${field}_rowspan`])
}
if (item[`_${field}_colspan`]) {
colspan_ = Utils.sprintf(' colspan="%s"', item[`_${field}_colspan`])
}
if (item[`_${field}_title`]) {
title_ = Utils.sprintf(' title="%s"', item[`_${field}_title`])
}
cellStyle = Utils.calculateObjectValue(this.header,
this.header.cellStyles[j], [value_, item, i, field], cellStyle)
if (cellStyle.classes) {
class_ = ` class="${cellStyle.classes}"`
}
if (cellStyle.css) {
const csses_ = []
for (const [key, value] of Object.entries(cellStyle.css)) {
csses_.push(`${key}: ${value}`)
}
style_ = ` style="${csses_.concat(this.header.styles[j]).join('; ')}"`
}
value = Utils.calculateObjectValue(column,
this.header.formatters[j], [value_, item, i, field], value_)
if (item[`_${field}_data`] && !Utils.isEmptyObject(item[`_${field}_data`])) {
for (const [k, v] of Object.entries(item[`_${field}_data`])) {
// ignore data-index
if (k === 'index') {
return
}
data_ += ` data-${k}="${v}"`
}
}
if (column.checkbox || column.radio) {
type = column.checkbox ? 'checkbox' : type
type = column.radio ? 'radio' : type
const c = column['class'] || ''
const isChecked = (value === true || value_ || (value && value.checked)) && value !== false
const isDisabled = !column.checkboxEnabled || (value && value.disabled)
text = [
this.options.cardView
? `<div class="card-view ${c}">`
: `<td class="bs-checkbox ${c}"${class_}${style_}>`,
`<label>
<input
data-index="${i}"
name="${this.options.selectItemName}"
type="${type}"
${Utils.sprintf('value="%s"', item[this.options.idField])}
${Utils.sprintf('checked="%s"', isChecked ? 'checked' : undefined)}
${Utils.sprintf('disabled="%s"', isDisabled ? 'disabled' : undefined)} />
<span></span>
</label>`,
this.header.formatters[j] && typeof value === 'string' ? value : '',
this.options.cardView ? '</div>' : '</td>'
].join('')
item[this.header.stateField] = value === true || (!!value_ || (value && value.checked))
} else {
value = typeof value === 'undefined' || value === null
? this.options.undefinedText : value
if (this.options.cardView) {
const cardTitle = this.options.showHeader
? `<span class="card-view-title"${style_}>${Utils.getFieldTitle(this.columns, field)}</span>` : ''
text = `<div class="card-view">${cardTitle}<span class="card-view-value">${value}</span></div>`
if (this.options.smartDisplay && value === '') {
text = '<div class="card-view"></div>'
}
} else {
text = `<td${id_}${class_}${style_}${data_}${rowspan_}${colspan_}${title_}>${value}</td>`
}
}
html.push(text)
})
if (this.options.cardView) {
html.push('</div></td>')
}
html.push('</tr>')
return html.join('')
}
initBody (fixedScroll) {
const data = this.getData()
this.trigger('pre-body', data)
this.$body = this.$el.find('>tbody')
if (!this.$body.length) {
this.$body = $('<tbody></tbody>').appendTo(this.$el)
}
// Fix #389 Bootstrap-table-flatJSON is not working
if (!this.options.pagination || this.options.sidePagination === 'server') {
this.pageFrom = 1
this.pageTo = data.length
}
const rows = []
const trFragments = $(document.createDocumentFragment())
let hasTr = false
for (let i = this.pageFrom - 1; i < this.pageTo; i++) {
const item = data[i]
const tr = this.initRow(item, i, data, trFragments)
hasTr = hasTr || !!tr
if (tr && typeof tr === 'string') {
if (!this.options.virtualScroll) {
trFragments.append(tr)
} else {
rows.push(tr)
}
}
}
// show no records
if (!hasTr) {
this.$body.html(`<tr class="no-records-found">${Utils.sprintf('<td colspan="%s">%s</td>',
this.$header.find('th').length,
this.options.formatNoMatches())}</tr>`)
} else {
if (!this.options.virtualScroll) {
this.$body.html(trFragments)
} else {
if (this.virtualScroll) {
this.virtualScroll.destroy()
}
this.virtualScroll = new VirtualScroll({
rows,
scrollEl: this.$tableBody[0],
contentEl: this.$body[0],
itemHeight: this.options.virtualScrollItemHeight,
callback: () => {
this.fitHeader()
}
})
}
}
if (!fixedScroll) {
this.scrollTo(0)
}
// click to select by column
this.$body.find('> tr[data-index] > td').off('click dblclick').on('click dblclick', e => {
const $td = $(e.currentTarget)
const $tr = $td.parent()
const $cardViewArr = $(e.target).parents('.card-views').children()
const $cardViewTarget = $(e.target).parents('.card-view')
const rowIndex = $tr.data('index')
const item = this.data[rowIndex]
const index = this.options.cardView ? $cardViewArr.index($cardViewTarget) : $td[0].cellIndex
const fields = this.getVisibleFields()
const field = fields[this.options.detailView && !this.options.cardView ? index - 1 : index]
const column = this.columns[this.fieldsColumnsIndex[field]]
const value = Utils.getItemField(item, field, this.options.escape)
if ($td.find('.detail-icon').length) {
return
}
this.trigger(e.type === 'click' ? 'click-cell' : 'dbl-click-cell', field, value, item, $td)
this.trigger(e.type === 'click' ? 'click-row' : 'dbl-click-row', item, $tr, field)
// if click to select - then trigger the checkbox/radio click
if (
e.type === 'click' &&
this.options.clickToSelect &&
column.clickToSelect &&
!Utils.calculateObjectValue(this.options, this.options.ignoreClickToSelectOn, [e.target])
) {
const $selectItem = $tr.find(Utils.sprintf('[name="%s"]', this.options.selectItemName))
if ($selectItem.length) {
$selectItem[0].click()
}
}
if (e.type === 'click' && this.options.detailViewByClick) {
this.toggleDetailView(rowIndex, this.header.detailFormatters[index])
}
}).off('mousedown').on('mousedown', e => {
// https://github.com/jquery/jquery/issues/1741
this.multipleSelectRowCtrlKey = e.ctrlKey || e.metaKey
this.multipleSelectRowShiftKey = e.shiftKey
})
this.$body.find('> tr[data-index] > td > .detail-icon').off('click').on('click', e => {
e.preventDefault()
this.toggleDetailView($(e.currentTarget).parent().parent().data('index'))
return false
})
this.$selectItem = this.$body.find(Utils.sprintf('[name="%s"]', this.options.selectItemName))
this.$selectItem.off('click').on