2021-03-10 09:43:36 +01:00
|
|
|
<?php
|
|
|
|
use Cake\Utility\Inflector;
|
|
|
|
|
|
|
|
$filteringForm = $this->Bootstrap->table(
|
|
|
|
[
|
|
|
|
'small' => true,
|
|
|
|
'striped' => false,
|
|
|
|
'hover' => false,
|
|
|
|
'tableClass' => ['indexFilteringTable'],
|
|
|
|
],
|
|
|
|
[
|
|
|
|
'fields' => [
|
|
|
|
__('Field'),
|
|
|
|
__('Operator'),
|
|
|
|
[
|
|
|
|
'labelHtml' => sprintf('%s %s',
|
|
|
|
__('Value'),
|
2021-03-11 09:18:12 +01:00
|
|
|
sprintf('<sup class="fa fa-info" title="%s"><sup>', __('Supports strict match and LIKE match with the `%` character. Example: `%.com`'))
|
2021-03-10 09:43:36 +01:00
|
|
|
)
|
|
|
|
],
|
|
|
|
__('Action')
|
|
|
|
],
|
|
|
|
'items' => []
|
|
|
|
]);
|
|
|
|
|
|
|
|
|
|
|
|
echo $this->Bootstrap->modal([
|
|
|
|
'title' => __('Filtering options for {0}', Inflector::singularize($this->request->getParam('controller'))),
|
|
|
|
'size' => 'lg',
|
|
|
|
'type' => 'confirm',
|
|
|
|
'bodyHtml' => $filteringForm,
|
|
|
|
'confirmText' => __('Filter'),
|
2021-03-15 22:47:13 +01:00
|
|
|
'confirmFunction' => 'filterIndex(clicked)'
|
2021-03-10 09:43:36 +01:00
|
|
|
]);
|
|
|
|
?>
|
|
|
|
|
|
|
|
<script>
|
|
|
|
$(document).ready(() => {
|
|
|
|
const $filteringTable = $('table.indexFilteringTable')
|
|
|
|
initFilteringTable($filteringTable)
|
|
|
|
})
|
|
|
|
|
|
|
|
function filterIndex(clicked) {
|
|
|
|
const controller = '<?= $this->request->getParam('controller') ?>';
|
|
|
|
const action = 'index';
|
|
|
|
const $clicked = $(clicked)
|
|
|
|
const $tbody = $clicked.closest('div.modal-content').find('table.indexFilteringTable tbody')
|
|
|
|
const $rows = $tbody.find('tr:not(#controlRow)')
|
|
|
|
const activeFilters = {}
|
|
|
|
$rows.each(function() {
|
|
|
|
const rowData = getDataFromRow($(this))
|
|
|
|
let fullFilter = rowData['name']
|
|
|
|
if (rowData['operator'] == '!=') {
|
|
|
|
fullFilter += ' !='
|
|
|
|
}
|
|
|
|
activeFilters[fullFilter] = rowData['value']
|
|
|
|
})
|
|
|
|
const searchParam = (new URLSearchParams(activeFilters)).toString();
|
|
|
|
const url = `/${controller}/${action}?${searchParam}`
|
|
|
|
|
|
|
|
const randomValue = getRandomValue()
|
|
|
|
UI.reload(url, $(`#table-container-${randomValue}`), $(`#table-container-${randomValue} table.table`), [{
|
|
|
|
node: $(`#toggleFilterButton-${randomValue}`),
|
|
|
|
config: {}
|
|
|
|
}])
|
|
|
|
}
|
|
|
|
|
|
|
|
function initFilteringTable($filteringTable) {
|
|
|
|
const $controlRow = $filteringTable.find('#controlRow')
|
|
|
|
$filteringTable.find('tbody').empty()
|
|
|
|
addControlRow($filteringTable)
|
|
|
|
const randomValue = getRandomValue()
|
|
|
|
const activeFilters = $(`#toggleFilterButton-${randomValue}`).data('activeFilters')
|
|
|
|
for (let [field, value] of Object.entries(activeFilters)) {
|
|
|
|
const fieldParts = field.split(' ')
|
|
|
|
let operator = '='
|
|
|
|
if (fieldParts.length == 2 && fieldParts[1] == '!=') {
|
|
|
|
operator = '!='
|
|
|
|
field = fieldParts[0]
|
|
|
|
} else if (fieldParts.length > 2) {
|
|
|
|
console.error('Field contains multiple spaces. ' + field)
|
|
|
|
}
|
|
|
|
addFilteringRow($filteringTable, field, value, operator)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function addControlRow($filteringTable) {
|
|
|
|
const availableFilters = <?= json_encode($filters) ?>;
|
|
|
|
const $selectField = $('<select/>').addClass('fieldSelect custom-select custom-select-sm')
|
|
|
|
availableFilters.forEach(filter => {
|
|
|
|
$selectField.append($('<option/>').text(filter))
|
|
|
|
});
|
|
|
|
const $selectOperator = $('<select/>').addClass('fieldOperator custom-select custom-select-sm')
|
|
|
|
.append([
|
|
|
|
$('<option/>').text('=').val('='),
|
|
|
|
$('<option/>').text('!=').val('!='),
|
|
|
|
])
|
|
|
|
const $row = $('<tr/>').attr('id', 'controlRow')
|
|
|
|
.append(
|
|
|
|
$('<td/>').append($selectField),
|
|
|
|
$('<td/>').append($selectOperator),
|
|
|
|
$('<td/>').append(
|
|
|
|
$('<input>').attr('type', 'text').addClass('fieldValue form-control form-control-sm')
|
|
|
|
),
|
|
|
|
$('<td/>').append(
|
|
|
|
$('<button/>').attr('type', 'button').addClass('btn btn-sm btn-primary')
|
|
|
|
.append($('<span/>').addClass('fa fa-plus'))
|
|
|
|
.click(addFiltering)
|
|
|
|
)
|
|
|
|
)
|
|
|
|
$filteringTable.append($row)
|
|
|
|
}
|
|
|
|
|
|
|
|
function addFilteringRow($filteringTable, field, value, operator) {
|
|
|
|
const $selectOperator = $('<select/>').addClass('fieldOperator custom-select custom-select-sm')
|
|
|
|
.append([
|
|
|
|
$('<option/>').text('=').val('='),
|
|
|
|
$('<option/>').text('!=').val('!='),
|
|
|
|
]).val(operator)
|
|
|
|
const $row = $('<tr/>')
|
|
|
|
.append(
|
|
|
|
$('<td/>').text(field).addClass('fieldName').data('fieldName', field),
|
|
|
|
$('<td/>').append($selectOperator),
|
|
|
|
$('<td/>').append(
|
|
|
|
$('<input>').attr('type', 'text').addClass('fieldValue form-control form-control-sm').val(value)
|
|
|
|
),
|
|
|
|
$('<td/>').append(
|
|
|
|
$('<button/>').attr('type', 'button').addClass('btn btn-sm btn-danger')
|
|
|
|
.append($('<span/>').addClass('fa fa-trash'))
|
|
|
|
.click(removeSelf)
|
|
|
|
)
|
|
|
|
)
|
|
|
|
$filteringTable.append($row)
|
|
|
|
const $controlRow = $filteringTable.find('#controlRow')
|
|
|
|
disableOptionFromSelect($controlRow, field)
|
|
|
|
}
|
|
|
|
|
|
|
|
function addFiltering() {
|
|
|
|
const $table = $(this).closest('table.indexFilteringTable')
|
|
|
|
const $controlRow = $table.find('#controlRow')
|
|
|
|
const field = $controlRow.find('select.fieldSelect').val()
|
|
|
|
const value = $controlRow.find('input.fieldValue').val()
|
|
|
|
const operator = $controlRow.find('input.fieldOperator').val()
|
|
|
|
addFilteringRow($table, field, value, operator)
|
|
|
|
$controlRow.find('input.fieldValue').val('')
|
|
|
|
$controlRow.find('select.fieldSelect').val('')
|
|
|
|
}
|
|
|
|
|
|
|
|
function removeSelf() {
|
|
|
|
const $row = $(this).closest('tr')
|
|
|
|
const $controlRow = $row.closest('table.indexFilteringTable').find('#controlRow')
|
|
|
|
const field = $row.data('fieldName')
|
|
|
|
$row.remove()
|
|
|
|
enableOptionFromSelect($controlRow, field)
|
|
|
|
}
|
|
|
|
|
|
|
|
function disableOptionFromSelect($controlRow, optionName) {
|
|
|
|
$controlRow.find('select.fieldSelect option').each(function() {
|
|
|
|
const $option = $(this)
|
|
|
|
if ($option.text() == optionName) {
|
|
|
|
$option.prop('disabled', true)
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
function enableOptionFromSelect($controlRow, optionName) {
|
|
|
|
$controlRow.find('select.fieldSelect option').each(function() {
|
|
|
|
const $option = $(this)
|
|
|
|
if ($option.text() == optionName) {
|
|
|
|
$option.prop('disabled', false)
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
function getDataFromRow($row) {
|
|
|
|
const rowData = {};
|
|
|
|
rowData['name'] = $row.find('td.fieldName').data('fieldName')
|
|
|
|
rowData['operator'] = $row.find('select.fieldOperator').val()
|
|
|
|
rowData['value'] = $row.find('input.fieldValue').val()
|
|
|
|
return rowData
|
|
|
|
}
|
|
|
|
|
|
|
|
function getRandomValue() {
|
|
|
|
const $container = $('div[id^="table-container-"]')
|
|
|
|
const randomValue = $container.attr('id').split('-')[2]
|
|
|
|
return randomValue
|
|
|
|
}
|
|
|
|
</script>
|