grid filter

This commit is contained in:
Tsanie Lily 2023-04-24 17:38:48 +08:00
parent 4096068a62
commit 811467bc7a
3 changed files with 146 additions and 33 deletions

View File

@ -8,7 +8,7 @@
overflow: visible; overflow: visible;
& { & {
--hover-bg-color: lightyellow; --cell-hover-bg-color: lightyellow;
--header-border-color: #adaba9; --header-border-color: #adaba9;
--header-bg-color: #fafafa; --header-bg-color: #fafafa;
--header-fore-color: #000; --header-fore-color: #000;
@ -46,6 +46,8 @@
--header-filter-padding: 4px 26px 4px 8px; --header-filter-padding: 4px 26px 4px 8px;
--spacing-s: 4px; --spacing-s: 4px;
--spacing-cell: 6px 4px 6px 8px; --spacing-cell: 6px 4px 6px 8px;
--filter-line-height: 30px;
--filter-item-padding: 0 4px;
} }
@include outline(); @include outline();
@ -369,7 +371,7 @@
position: absolute; position: absolute;
line-height: var(--line-height); line-height: var(--line-height);
padding: var(--spacing-cell); padding: var(--spacing-cell);
background-color: var(--hover-bg-color); background-color: var(--cell-hover-bg-color);
white-space: pre; white-space: pre;
display: flex; display: flex;
align-items: center; align-items: center;
@ -449,12 +451,72 @@
cursor: text; cursor: text;
} }
} }
>.filter-item-list {
flex: 1 1 auto;
overflow-y: auto;
overflow-x: hidden;
position: relative;
user-select: none;
@include scrollbar();
>.filter-content {
position: absolute;
width: 100%;
}
.filter-item {
width: 100%;
white-space: nowrap;
overflow: hidden;
text-overflow: ellipsis;
box-sizing: border-box;
padding: var(--filter-item-padding);
&:hover {
background-color: var(--hover-bg-color);
}
.ui-check-wrapper {
height: var(--filter-line-height);
display: flex;
.ui-check-inner+* {
font-size: var(--font-smaller-size);
}
}
}
}
>.filter-function {
display: flex;
justify-content: flex-end;
padding: 4px;
>button {
box-sizing: border-box;
margin-right: 10px;
min-width: 40px;
height: var(--filter-line-height);
border: none;
background-color: transparent;
cursor: pointer;
border-radius: 0;
transition: background-color .12s ease;
@include outline();
&:hover {
background-color: var(--hover-bg-color);
}
}
}
} }
} }
@media (prefers-color-scheme: dark) { @media (prefers-color-scheme: dark) {
.ui-grid { .ui-grid {
--hover-bg-color: yellow; --cell-hover-bg-color: yellow;
--header-border-color: #525456; --header-border-color: #525456;
--header-bg-color: #050505; --header-bg-color: #050505;
--header-fore-color: #fff; --header-fore-color: #fff;

View File

@ -1,5 +1,5 @@
import '../css/grid.scss'; import '../css/grid.scss';
import { global, isPositive, isMobile, throttle, truncate, distinct } from "../../utility"; import { global, isPositive, isMobile, throttle, truncate } from "../../utility";
import { r } from "../../utility/lgres"; import { r } from "../../utility/lgres";
import { createElement } from "../../functions"; import { createElement } from "../../functions";
import { createIcon } from "../icon"; import { createIcon } from "../icon";
@ -125,12 +125,17 @@ class Grid {
} }
list = list.map(i => { return { values: i } }); list = list.map(i => { return { values: i } });
this.#source = list; this.#source = list;
this.#refreshSource(list);
}
#refreshSource(list) {
list ??= this.#source;
if (this.#colAttrs.__filtered === true) { if (this.#colAttrs.__filtered === true) {
this.#currentSource = list.filter(it => { this.#currentSource = list.filter(it => {
for (let col of this.columns) { for (let col of this.columns) {
const f = this.#get(col.key, 'filter'); const f = this.#get(col.key, 'filter');
if (Array.isArray(f)) { if (Array.isArray(f)) {
const v = this.#getItemValue(it, col.key, col.filter); const v = this.#getItemValue(it.values, col.key, col.filter);
if (f.indexOf(v) < 0) { if (f.indexOf(v) < 0) {
return false; return false;
} }
@ -375,8 +380,8 @@ class Grid {
direction = 1; direction = 1;
} }
comparer = (a, b) => { comparer = (a, b) => {
a = this.#getItemValue(a, col.key, col.filter); a = this.#getItemValue(a.values, col.key, col.filter);
b = this.#getItemValue(b, col.key, col.filter); b = this.#getItemValue(b.values, col.key, col.filter);
if (a == null && typeof b === 'number') { if (a == null && typeof b === 'number') {
a = 0; a = 0;
} else if (typeof a === 'number' && b == null) { } else if (typeof a === 'number' && b == null) {
@ -936,7 +941,7 @@ class Grid {
if (typeof filter === 'function') { if (typeof filter === 'function') {
value = filter(item); value = filter(item);
} else { } else {
value = item.values[key]; value = item[key];
} }
return value?.value ?? value; return value?.value ?? value;
} }
@ -1005,7 +1010,8 @@ class Grid {
document.addEventListener('mousedown', close); document.addEventListener('mousedown', close);
const panel = createElement('div', 'filter-panel'); const panel = createElement('div', 'filter-panel');
panel.addEventListener('mousedown', e => e.stopPropagation()); panel.addEventListener('mousedown', e => e.stopPropagation());
const th = e.currentTarget.parentElement; const filter = e.currentTarget;
const th = filter.parentElement;
const width = th.offsetWidth; const width = th.offsetWidth;
panel.style.top = `${th.offsetHeight}px`; panel.style.top = `${th.offsetHeight}px`;
panel.style.left = (th.offsetLeft + (width > FilterPanelWidth ? width - FilterPanelWidth : 0)) + 'px'; panel.style.left = (th.offsetLeft + (width > FilterPanelWidth ? width - FilterPanelWidth : 0)) + 'px';
@ -1044,7 +1050,18 @@ class Grid {
} else if (typeof col.filterSource === 'function') { } else if (typeof col.filterSource === 'function') {
array = col.filterSource.call(this, col); array = col.filterSource.call(this, col);
} else { } else {
array = distinct(this.#currentSource, col.key, col.filter) const dict = Object.create(null);
for (let item of this.#source) {
const val = this.#getItemValue(item.values, col.key, col.filter);
if (!Object.hasOwnProperty.call(dict, val)) {
const v = item.values[col.key];
dict[val] = {
value: val,
displayValue: typeof col.filter === 'function' ? col.filter(item.values) : v?.displayValue ?? v
};
}
}
array = Object.values(dict)
.sort((a, b) => { .sort((a, b) => {
a = a?.value ?? a; a = a?.value ?? a;
b = b?.value ?? b; b = b?.value ?? b;
@ -1062,7 +1079,7 @@ class Grid {
}; };
}); });
this.#fillFilterList(col, itemlist, array, itemall); this.#fillFilterList(col, itemlist, array, itemall);
// TODO: check status itemall.querySelector('input').checked = ![...itemlist.querySelectorAll('.filter-content input')].some(i => !i.checked);
panel.appendChild(itemlist); panel.appendChild(itemlist);
if (searchbox != null) { if (searchbox != null) {
searchbox.addEventListener('input', e => { searchbox.addEventListener('input', e => {
@ -1074,16 +1091,52 @@ class Grid {
this.#fillFilterList(col, itemlist, items, itemall); this.#fillFilterList(col, itemlist, items, itemall);
}); });
} }
// function
const functions = createElement('div', 'filter-function');
functions.append(
createElement('button', ok => {
ok.innerText = this.langs.ok;
ok.addEventListener('click', () => {
const array = this.#get(col.key, 'filterSource').filter(i => i.__checked !== false);
if (typeof col.onFilterOk === 'function') {
col.onFilterOk.call(this, col, array);
} else {
this.#set(col.key, 'filter', array.map(a => a.value));
}
this.#colAttrs.__filtered = true;
this.#refreshSource();
if (typeof col.onFiltered === 'function') {
col.onFiltered.call(this, col);
}
filter.classList.add('active');
this.#onCloseFilter();
});
}),
createElement('button', reset => {
reset.innerText = this.langs.reset;
reset.addEventListener('click', () => {
this.#set(col.key, 'filter', null);
// TODO: change __filtered
this.#refreshSource();
if (typeof col.onFiltered === 'function') {
col.onFiltered.call(this, col);
}
filter.classList.remove('active');
this.#onCloseFilter();
});
})
);
panel.appendChild(functions);
this.#el.appendChild(panel); this.#el.appendChild(panel);
setTimeout(() => panel.classList.add('active'), 0); setTimeout(() => panel.classList.add('active'), 0);
this.#colAttrs.__filtering = e.currentTarget; this.#colAttrs.__filtering = filter;
e.currentTarget.classList.add('hover'); filter.classList.add('hover');
} }
#fillFilterList(col, list, array, all) { #fillFilterList(col, list, array, all) {
list.querySelector('.filter-holder').remove(); list.querySelector('.filter-holder')?.remove();
list.querySelector('.filter-content').remove(); list.querySelector('.filter-content')?.remove();
const rowHeight = this.filterRowHeight; const rowHeight = this.filterRowHeight;
const height = array.length * rowHeight; const height = array.length * rowHeight;
this.#set(col.key, 'filterHeight', height); this.#set(col.key, 'filterHeight', height);
@ -1094,17 +1147,28 @@ class Grid {
this.#set(col.key, 'filterSource', array); this.#set(col.key, 'filterSource', array);
const filter = this.#get(col.key, 'filter'); const filter = this.#get(col.key, 'filter');
for (let item of array) { for (let item of array) {
item.__checked = !Array.isArray(filter) || filter.indexOf(item.value ?? item); item.__checked = !Array.isArray(filter) || filter.indexOf(item.value ?? item) >= 0;
} }
if (array.length > 12) { if (array.length > 12) {
array = array.slice(0, 12); array = array.slice(0, 12);
} }
this.#doFillFilterList(col, content, array, all); this.#doFillFilterList(content, array, all);
list.append(holder, content); list.append(holder, content);
} }
#doFillFilterList(col, content, array, all) { #doFillFilterList(content, array, all) {
for (let item of array) {
const div = createElement('div', 'filter-item');
div.appendChild(createCheckbox({
checked: item.__checked,
label: item?.displayValue ?? item,
onchange: e => {
item.__checked = e.target.checked;
all.querySelector('input').checked = ![...content.querySelectorAll('input')].some(i => !i.checked);
}
}));
content.appendChild(div);
}
} }
#onFilterScroll(col, list, top) { #onFilterScroll(col, list, top) {
@ -1132,7 +1196,7 @@ class Grid {
} }
const content = list.querySelector('.filter-content'); const content = list.querySelector('.filter-content');
content.replaceChildren(); content.replaceChildren();
this.#doFillFilterList(col, content, array, list.querySelector('.filter-all>input')); this.#doFillFilterList(content, array, list.querySelector('.filter-all>input'));
content.style.top = `${top + rowHeight}px`; content.style.top = `${top + rowHeight}px`;
} }
} }

View File

@ -39,18 +39,6 @@ function truncate(v) {
return (v > 0 ? Math.floor : Math.ceil)(v); return (v > 0 ? Math.floor : Math.ceil)(v);
} }
function distinct(array, key, filter) {
const dict = Object.create(null);
for (let item of array) {
const v = typeof filter === 'function' ? filter(item) : item[key];
const val = v?.value ?? v;
if (!Object.prototype.hasOwnProperty.call(dict, val)) {
dict[val] = v;
}
}
return Object.values(dict);
}
function isEmail(text) { function isEmail(text) {
return /^\w[-\w.+]*@\w+([-.]\w+)*\.\w+([-.]\w+)*$/.test(text); return /^\w[-\w.+]*@\w+([-.]\w+)*\.\w+([-.]\w+)*$/.test(text);
} }
@ -87,7 +75,6 @@ export {
throttle, throttle,
debounce, debounce,
truncate, truncate,
distinct,
isEmail, isEmail,
isPhone isPhone
} }