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,455 lines (1,241 loc) • 93.1 kB
JavaScript
/**
* @author zhixin wen <wenzhixin2010@gmail.com>
* version: 1.16.0
* 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 opts = this.options
this.constants = Constants.CONSTANTS
this.constants.theme = $.fn.bootstrapTable.theme
const buttonsPrefix = opts.buttonsPrefix ? `${opts.buttonsPrefix}-` : ''
this.constants.buttonsClass = [
opts.buttonsPrefix,
buttonsPrefix + opts.buttonsClass,
Utils.sprintf(`${buttonsPrefix}%s`, opts.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 (this.options.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: []
}
Utils.updateFieldGroup(this.options.columns)
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 = 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' && !column.visible) {
return
}
const headerStyle = Utils.calculateObjectValue(null, this.options.headerStyle, [column])
const csses = []
let classes = ''
if (headerStyle && headerStyle.css) {
for (const [key, value] of Object.entries(headerStyle.css)) {
csses.push(`${key}: ${value}`)
}
}
if (headerStyle && headerStyle.classes) {
classes = Utils.sprintf(' class="%s"', column['class'] ?
[column['class'], headerStyle.classes].join(' ') : headerStyle.classes)
}
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'] || '')
: classes || class_,
Utils.sprintf(' style="%s"', halign + style + csses.join('; ')),
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
}
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 = Utils.getResizeEventName(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, () => this.resetView())
}
this.$selectAll = this.$header.find('[name="btSelectAll"]')
this.$selectAll.off('click').on('click', e => {
e.stopPropagation()
const checked = $(e.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,
a._position, b._position)
})
}
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.serverSort) {
this.options.pageNumber = 1
this.initServer(this.options.silentSort)
return
}
this.initSort()
this.initBody()
}
initToolbar () {
const opts = this.options
let html = []
let timeoutId = 0
let $keepOpen
let $search
let switchableCount = 0
if (this.$toolbar.find('.bs-bars').children().length) {
$('body').append($(opts.toolbar))
}
this.$toolbar.html('')
if (typeof opts.toolbar === 'string' || typeof opts.toolbar === 'object') {
$(Utils.sprintf('<div class="bs-bars %s-%s"></div>', this.constants.classes.pull, opts.toolbarAlign))
.appendTo(this.$toolbar)
.append($(opts.toolbar))
}
// showColumns, showToggle, showRefresh
html = [`<div class="${[
'columns',
`columns-${opts.buttonsAlign}`,
this.constants.classes.buttonsGroup,
`${this.constants.classes.pull}-${opts.buttonsAlign}`
].join(' ')}">`]
if (typeof opts.icons === 'string') {
opts.icons = Utils.calculateObjectValue(null, opts.icons)
}
const buttonsHtml = {
paginationSwitch: `<button class="${this.constants.buttonsClass}" type="button" name="paginationSwitch"
aria-label="Pagination Switch" title="${opts.formatPaginationSwitch()}">
${opts.showButtonIcons ? Utils.sprintf(this.constants.html.icon, opts.iconsPrefix, opts.icons.paginationSwitchDown) : ''}
${opts.showButtonText ? opts.formatPaginationSwitchUp() : ''}
</button>`,
refresh: `<button class="${this.constants.buttonsClass}" type="button" name="refresh"
aria-label="Refresh" title="${opts.formatRefresh()}">
${opts.showButtonIcons ? Utils.sprintf(this.constants.html.icon, opts.iconsPrefix, opts.icons.refresh) : ''}
${opts.showButtonText ? opts.formatRefresh() : ''}
</button>`,
toggle: `<button class="${this.constants.buttonsClass}" type="button" name="toggle"
aria-label="Toggle" title="${opts.formatToggle()}">
${opts.showButtonIcons ? Utils.sprintf(this.constants.html.icon, opts.iconsPrefix, opts.icons.toggleOff) : '' }
${opts.showButtonText ? opts.formatToggleOn() : ''}
</button>`,
fullscreen: `<button class="${this.constants.buttonsClass}" type="button" name="fullscreen"
aria-label="Fullscreen" title="${opts.formatFullscreen()}">
${opts.showButtonIcons ? Utils.sprintf(this.constants.html.icon, opts.iconsPrefix, opts.icons.fullscreen) : '' }
${opts.showButtonText ? opts.formatFullscreen() : ''}
</button>`,
columns: (() => {
const html = []
html.push(`<div class="keep-open ${this.constants.classes.buttonsDropdown}" title="${opts.formatColumns()}">
<button class="${this.constants.buttonsClass} dropdown-toggle" type="button" data-toggle="dropdown"
aria-label="Columns" title="${opts.formatColumns()}">
${opts.showButtonIcons ? Utils.sprintf(this.constants.html.icon, opts.iconsPrefix, opts.icons.columns) : ''}
${opts.showButtonText ? opts.formatColumns() : ''}
${this.constants.html.dropdownCaret}
</button>
${this.constants.html.toolbarDropdown[0]}`)
if (opts.showColumnsSearch) {
html.push(
Utils.sprintf(this.constants.html.toolbarDropdownItem,
Utils.sprintf('<input type="text" class="%s" id="columnsSearch" placeholder="%s" autocomplete="off">', this.constants.classes.input, opts.formatSearch())
)
)
html.push(this.constants.html.toolbarDropdownSeparator)
}
if (opts.showColumnsToggleAll) {
const allFieldsVisible = this.getVisibleColumns().length === this.columns.filter(column => !this.isSelectionColumn(column)).length
html.push(
Utils.sprintf(this.constants.html.toolbarDropdownItem,
Utils.sprintf('<input type="checkbox" class="toggle-all" %s> <span>%s</span>',
allFieldsVisible ? 'checked="checked"' : '', opts.formatColumnsToggleAll())
)
)
html.push(this.constants.html.toolbarDropdownSeparator)
}
let visibleColumns = 0
this.columns.forEach((column, i) => {
if (column.visible) {
visibleColumns++
}
})
this.columns.forEach((column, i) => {
if (this.isSelectionColumn(column)) {
return
}
if (opts.cardView && !column.cardVisible) {
return
}
const checked = column.visible ? ' checked="checked"' : ''
const disabled = (visibleColumns <= this.options.minimumCountColumns) && checked ? ' disabled="disabled"' : ''
if (column.switchable) {
html.push(Utils.sprintf(this.constants.html.toolbarDropdownItem,
Utils.sprintf('<input type="checkbox" data-field="%s" value="%s"%s%s> <span>%s</span>',
column.field, i, checked, disabled, column.title)))
switchableCount++
}
})
html.push(this.constants.html.toolbarDropdown[1], '</div>')
return html.join('')
})()
}
if (typeof opts.buttonsOrder === 'string') {
opts.buttonsOrder = opts.buttonsOrder.replace(/\[|\]| |'/g, '').toLowerCase().split(',')
}
for (const button of opts.buttonsOrder) {
if (opts['show' + button.charAt(0).toUpperCase() + button.substring(1)]) {
html.push(buttonsHtml[button])
}
}
html.push('</div>')
// Fix #188: this.showToolbar is for extensions
if (this.showToolbar || html.length > 2) {
this.$toolbar.append(html.join(''))
}
if (opts.showPaginationSwitch) {
this.$toolbar.find('button[name="paginationSwitch"]')
.off('click').on('click', () => this.togglePagination())
}
if (opts.showFullscreen) {
this.$toolbar.find('button[name="fullscreen"]')
.off('click').on('click', () => this.toggleFullscreen())
}
if (opts.showRefresh) {
this.$toolbar.find('button[name="refresh"]')
.off('click').on('click', () => this.refresh())
}
if (opts.showToggle) {
this.$toolbar.find('button[name="toggle"]')
.off('click').on('click', () => {
this.toggleView()
})
}
if (opts.showColumns) {
$keepOpen = this.$toolbar.find('.keep-open')
const $checkboxes = $keepOpen.find('input[type="checkbox"]:not(".toggle-all")')
const $toggleAll = $keepOpen.find('input[type="checkbox"].toggle-all')
if (switchableCount <= opts.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.filter(column => !this.isSelectionColumn(column)).length)
})
$toggleAll.off('click').on('click', ({currentTarget}) => {
this._toggleAllColumns($(currentTarget).prop('checked'))
})
if (opts.showColumnsSearch) {
const $columnsSearch = $keepOpen.find('#columnsSearch')
const $listItems = $keepOpen.find('.dropdown-item-marker')
$columnsSearch.on('keyup paste change', ({currentTarget}) => {
const $this = $(currentTarget)
const searchValue = $this.val().toLowerCase()
$listItems.show()
$checkboxes.each((i, el) => {
const $checkbox = $(el)
const $listItem = $checkbox.parents('.dropdown-item-marker')
const text = $listItem.text().toLowerCase()
if (!text.includes(searchValue)) {
$listItem.hide()
}
})
})
}
}
// Fix #4516: this.showSearchClearButton is for extensions
if (opts.search || this.showSearchClearButton) {
html = []
const showSearchButton = Utils.sprintf(this.constants.html.searchButton,
this.constants.buttonsClass,
opts.formatSearch(),
opts.showButtonIcons ? Utils.sprintf(this.constants.html.icon, opts.iconsPrefix, opts.icons.search) : '',
opts.showButtonText ? opts.formatSearch() : ''
)
const showSearchClearButton = Utils.sprintf(this.constants.html.searchClearButton,
this.constants.buttonsClass,
opts.formatClearSearch(),
opts.showButtonIcons ? Utils.sprintf(this.constants.html.icon, opts.iconsPrefix, opts.icons.clearSearch) : '',
opts.showButtonText ? opts.formatClearSearch() : ''
)
const searchInputHtml = `<input class="${this.constants.classes.input}
${Utils.sprintf(' %s%s', this.constants.classes.inputPrefix, opts.iconSize)}
search-input" type="text" placeholder="${opts.formatSearch()}" autocomplete="off">`
let searchInputFinalHtml = searchInputHtml
if (opts.showSearchButton || opts.showSearchClearButton) {
const buttonsHtml = (opts.showSearchButton ? showSearchButton : '') +
(opts.showSearchClearButton ? showSearchClearButton : '')
searchInputFinalHtml = opts.search ? Utils.sprintf(this.constants.html.inputGroup,
searchInputHtml, buttonsHtml) : buttonsHtml
}
html.push(Utils.sprintf(`
<div class="${this.constants.classes.pull}-${opts.searchAlign} search ${this.constants.classes.inputGroup}">
%s
</div>
`, searchInputFinalHtml))
this.$toolbar.append(html.join(''))
const $searchInput = this.$toolbar.find('.search input')
const handleInputEvent = () => {
const eventTriggers = `keyup drop blur ${Utils.isIEBrowser() ? 'mouseup' : ''}`
$searchInput.off(eventTriggers).on(eventTriggers, event => {
if (opts.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({currentTarget: event.currentTarget})
}, opts.searchTimeOut)
})
}
if (opts.showSearchButton) {
this.$toolbar.find('.search button[name=search]').off('click').on('click', event => {
clearTimeout(timeoutId) // doesn't matter if it's 0
timeoutId = setTimeout(() => {
this.onSearch({currentTarget: $searchInput})
}, opts.searchTimeOut)
})
if (opts.searchOnEnterKey) {
handleInputEvent()
}
} else {
handleInputEvent()
}
if (opts.showSearchClearButton) {
this.$toolbar.find('.search button[name=clearSearch]').click(() => {
this.resetSearch()
})
}
}
}
onSearch ({currentTarget, firedByInitSearchText} = {}, overwriteSearchText = true) {
if (currentTarget !== undefined && $(currentTarget).length && overwriteSearchText) {
const text = $(currentTarget).val().trim()
if (this.options.trimOnSearch && $(currentTarget).val() !== text) {
$(currentTarget).val(text)
}
if (this.searchText === text && text.length > 0) {
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, this.filterColumns])
return
}
const s = this.searchText && (this.fromHtml
? 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
}
this.initSort()
}
initPagination () {
const opts = this.options
if (!opts.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 = opts.pageList
if (typeof pageList === 'string') {
pageList = pageList.replace(/\[|\]| /g, '').toLowerCase().split(',')
}
pageList = pageList.map(value => {
if (typeof value === 'string') {
return value.toLowerCase() === opts.formatAllRows().toLowerCase() ||
['all', 'unlimited'].includes(value.toLowerCase())
? opts.formatAllRows() : +value
}
return value
})
if (opts.sidePagination !== 'server') {
opts.totalRows = data.length
}
this.totalPages = 0
if (opts.totalRows) {
if (opts.pageSize === opts.formatAllRows()) {
opts.pageSize = opts.totalRows
allSelected = true
}
this.totalPages = ~~((opts.totalRows - 1) / opts.pageSize) + 1
opts.totalPages = this.totalPages
}
if (this.totalPages > 0 && opts.pageNumber > this.totalPages) {
opts.pageNumber = this.totalPages
}
this.pageFrom = (opts.pageNumber - 1) * opts.pageSize + 1
this.pageTo = opts.pageNumber * opts.pageSize
if (this.pageTo > opts.totalRows) {
this.pageTo = opts.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 = opts.onlyInfoPagination ?
opts.formatDetailPagination(opts.totalRows) :
opts.formatShowingRows(this.pageFrom, this.pageTo, opts.totalRows, opts.totalNotFiltered)
html.push(`<div class="${this.constants.classes.pull}-${opts.paginationDetailHAlign} pagination-detail">
<span class="pagination-info">
${paginationInfo}
</span>`)
if (!opts.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 ? opts.formatAllRows() : opts.pageSize}
</span>
${this.constants.html.dropdownCaret}
</button>
${this.constants.html.pageDropdown[0]}`]
pageList.forEach((page, i) => {
if (!opts.smartDisplay || i === 0 || pageList[i - 1] < opts.totalRows) {
let active
if (allSelected) {
active = page === opts.formatAllRows() ? this.constants.classes.dropdownActive : ''
} else {
active = page === opts.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(opts.formatRecordsPerPage(pageNumber.join('')))
html.push('</span></div>')
html.push(`<div class="${this.constants.classes.pull}-${opts.paginationHAlign} pagination">`,
Utils.sprintf(this.constants.html.pagination[0], Utils.sprintf(' pagination-%s', opts.iconSize)),
Utils.sprintf(this.constants.html.paginationItem, ' page-pre', opts.formatSRPaginationPreText(), opts.paginationPreText))
if (this.totalPages < opts.paginationSuccessivelySize) {
from = 1
to = this.totalPages
} else {
from = opts.pageNumber - opts.paginationPagesBySide
to = from + (opts.paginationPagesBySide * 2)
}
if (opts.pageNumber < (opts.paginationSuccessivelySize - 1)) {
to = opts.paginationSuccessivelySize
}
if (opts.paginationSuccessivelySize > this.totalPages - from) {
from = from - (opts.paginationSuccessivelySize - (this.totalPages - from)) + 1
}
if (from < 1) {
from = 1
}
if (to > this.totalPages) {
to = this.totalPages
}
const middleSize = Math.round(opts.paginationPagesBySide / 2)
const pageItem = (i, classes = '') => Utils.sprintf(this.constants.html.paginationItem,
classes + (i === opts.pageNumber ? ` ${this.constants.classes.paginationActive}` : ''), opts.formatSRPaginationPageText(i), i)
if (from > 1) {
let max = opts.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 - opts.paginationPagesBySide * 2) > opts.paginationPagesBySide &&
opts.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 - (opts.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) > opts.paginationPagesBySide * 2 &&
opts.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', opts.formatSRPaginationNextText(), opts.paginationNextText))
html.push(this.constants.html.pagination[1], '</div>')
}
this.$pagination.html(html.join(''))
const dropupClass = ['bottom', 'both'].includes(opts.paginationVAlign) ?
` ${this.constants.classes.dropup}` : ''
this.$pagination.last().find('.page-list > span').addClass(dropupClass)
if (!opts.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 (opts.smartDisplay) {
if (pageList.length < 2 || opts.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 (!opts.paginationLoop) {
if (opts.pageNumber === 1) {
$pre.addClass('disabled')
}
if (opts.pageNumber === this.totalPages) {
$next.addClass('disabled')
}
}
if (allSelected) {
opts.pageSize = opts.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 = th