add grid, initial version
This commit is contained in:
parent
c44aaf5177
commit
6234154f33
@ -1,3 +1,10 @@
|
|||||||
|
@mixin inset($top, $right, $bottom, $left) {
|
||||||
|
top: $top;
|
||||||
|
right: $right;
|
||||||
|
bottom: $bottom;
|
||||||
|
left: $left;
|
||||||
|
}
|
||||||
|
|
||||||
@mixin scrollbar() {
|
@mixin scrollbar() {
|
||||||
&::-webkit-scrollbar {
|
&::-webkit-scrollbar {
|
||||||
width: 8px;
|
width: 8px;
|
||||||
|
250
css/grid.scss
Normal file
250
css/grid.scss
Normal file
@ -0,0 +1,250 @@
|
|||||||
|
@keyframes loading-spinner {
|
||||||
|
0% {
|
||||||
|
transform: rotate(0deg);
|
||||||
|
}
|
||||||
|
|
||||||
|
100% {
|
||||||
|
transform: rotate(359deg);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.grid {
|
||||||
|
position: relative;
|
||||||
|
box-sizing: border-box;
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
|
||||||
|
& {
|
||||||
|
--hover-bg-color: lightyellow;
|
||||||
|
--header-border-color: #adaba9;
|
||||||
|
--header-bg-color: #fafafa;
|
||||||
|
--header-fore-color: #000;
|
||||||
|
--cell-border-color: #f0f0f0;
|
||||||
|
--cell-fore-color: #333;
|
||||||
|
--dark-border-color: #666;
|
||||||
|
--split-border-color: #b3b3b3;
|
||||||
|
--dragger-bg-color: #fff;
|
||||||
|
--row-bg-color: #fff;
|
||||||
|
--row-active-bg-color: #fafafa;
|
||||||
|
--row-selected-bg-color: #e6f2fb;
|
||||||
|
--text-disabled-color: gray;
|
||||||
|
--loading-bg-color: hsla(0, 0%, 100%, .4);
|
||||||
|
--loading-fore-color: rgba(0, 0, 0, .2);
|
||||||
|
|
||||||
|
--font-size: .8125rem;
|
||||||
|
--line-height: 36px;
|
||||||
|
--header-line-height: 26px;
|
||||||
|
--text-indent: 8px;
|
||||||
|
|
||||||
|
--loading-size: 40px;
|
||||||
|
--loading-border-radius: 20px;
|
||||||
|
|
||||||
|
--arrow-size: 4px;
|
||||||
|
--split-width: 8px;
|
||||||
|
--dragger-size: 20px;
|
||||||
|
--dragger-opacity: .4;
|
||||||
|
--dragger-cursor-size: 4px;
|
||||||
|
--dragger-cursor-opacity: .6;
|
||||||
|
|
||||||
|
--header-padding: 4px 12px 4px 8px;
|
||||||
|
--spacing-s: 4px;
|
||||||
|
--spacing-cell: 5px 4px 5px 8px;
|
||||||
|
}
|
||||||
|
|
||||||
|
&:focus,
|
||||||
|
&:focus-visible {
|
||||||
|
outline: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
&,
|
||||||
|
input[type="text"] {
|
||||||
|
font-size: var(--font-size);
|
||||||
|
}
|
||||||
|
|
||||||
|
>.grid-sizer {
|
||||||
|
position: absolute;
|
||||||
|
white-space: nowrap;
|
||||||
|
font-weight: bold;
|
||||||
|
visibility: hidden;
|
||||||
|
}
|
||||||
|
|
||||||
|
>.grid-header {
|
||||||
|
width: 100%;
|
||||||
|
min-width: 100%;
|
||||||
|
margin: 0;
|
||||||
|
border-bottom: 1px solid var(--header-border-color);
|
||||||
|
background-color: var(--header-bg-color);
|
||||||
|
color: var(--header-fore-color);
|
||||||
|
user-select: none;
|
||||||
|
border-collapse: collapse;
|
||||||
|
border-spacing: 0;
|
||||||
|
table-layout: fixed;
|
||||||
|
position: relative;
|
||||||
|
|
||||||
|
th {
|
||||||
|
padding: 0;
|
||||||
|
margin: 0;
|
||||||
|
word-wrap: break-word;
|
||||||
|
white-space: normal;
|
||||||
|
position: relative;
|
||||||
|
|
||||||
|
>div {
|
||||||
|
line-height: var(--header-line-height);
|
||||||
|
min-height: var(--line-height);
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
padding: var(--header-padding);
|
||||||
|
box-sizing: border-box;
|
||||||
|
}
|
||||||
|
|
||||||
|
>.arrow {
|
||||||
|
width: 0;
|
||||||
|
height: 0;
|
||||||
|
top: 50%;
|
||||||
|
margin-top: calc(0px - var(--arrow-size) / 2);
|
||||||
|
right: calc(var(--arrow-size) / 2);
|
||||||
|
position: absolute;
|
||||||
|
|
||||||
|
&.asc {
|
||||||
|
border-bottom: var(--arrow-size) solid var(--dark-border-color);
|
||||||
|
}
|
||||||
|
|
||||||
|
&.desc {
|
||||||
|
border-top: var(--arrow-size) solid var(--dark-border-color);
|
||||||
|
}
|
||||||
|
|
||||||
|
&.asc,
|
||||||
|
&.desc {
|
||||||
|
border-left: var(--arrow-size) solid transparent;
|
||||||
|
border-right: var(--arrow-size) solid transparent;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
>.spliter {
|
||||||
|
position: absolute;
|
||||||
|
height: 100%;
|
||||||
|
top: 0;
|
||||||
|
right: calc(0px - var(--split-width) /2);
|
||||||
|
width: var(--split-width);
|
||||||
|
cursor: ew-resize;
|
||||||
|
z-index: 1;
|
||||||
|
|
||||||
|
&::after {
|
||||||
|
content: '';
|
||||||
|
height: 100%;
|
||||||
|
width: 1px;
|
||||||
|
display: block;
|
||||||
|
margin: 0 auto;
|
||||||
|
transition: background-color .12s ease;
|
||||||
|
}
|
||||||
|
|
||||||
|
&:hover::after {
|
||||||
|
background-color: var(--split-border-color);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
>.grid-body {
|
||||||
|
flex: 1 1 auto;
|
||||||
|
overflow: auto;
|
||||||
|
color: var(--cell-fore-color);
|
||||||
|
@include scrollbar();
|
||||||
|
|
||||||
|
.grid-body-content {
|
||||||
|
position: absolute;
|
||||||
|
min-width: 100%;
|
||||||
|
table-layout: fixed;
|
||||||
|
border-collapse: collapse;
|
||||||
|
border-spacing: 0;
|
||||||
|
|
||||||
|
>.grid-row {
|
||||||
|
line-height: var(--line-height);
|
||||||
|
white-space: nowrap;
|
||||||
|
background-color: var(--row-bg-color);
|
||||||
|
border-bottom: 1px solid var(--cell-border-color);
|
||||||
|
box-sizing: border-box;
|
||||||
|
|
||||||
|
&:hover {
|
||||||
|
background-color: var(--row-active-bg-color);
|
||||||
|
}
|
||||||
|
|
||||||
|
&.selected {
|
||||||
|
background-color: var(--row-selected-bg-color);
|
||||||
|
}
|
||||||
|
|
||||||
|
>td {
|
||||||
|
padding: 0;
|
||||||
|
overflow: hidden;
|
||||||
|
text-overflow: ellipsis;
|
||||||
|
white-space: pre;
|
||||||
|
|
||||||
|
>span {
|
||||||
|
padding: var(--spacing-cell);
|
||||||
|
}
|
||||||
|
|
||||||
|
>input[type="text"] {
|
||||||
|
border: none;
|
||||||
|
box-sizing: border-box;
|
||||||
|
height: calc(var(--line-height) + 2px);
|
||||||
|
width: 100%;
|
||||||
|
text-indent: var(--text-indent);
|
||||||
|
|
||||||
|
&:focus,
|
||||||
|
&:focus-visible {
|
||||||
|
outline: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
&:disabled {
|
||||||
|
color: var(--text-disabled-color);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.checkbox-wrapper .check-box-inner {
|
||||||
|
|
||||||
|
&,
|
||||||
|
>svg {
|
||||||
|
transition: none;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.grid-hover-holder {
|
||||||
|
box-sizing: border-box;
|
||||||
|
position: absolute;
|
||||||
|
line-height: var(--line-height);
|
||||||
|
padding: var(--spacing-cell);
|
||||||
|
background-color: var(--hover-bg-color);
|
||||||
|
white-space: pre;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
>.grid-loading {
|
||||||
|
position: absolute;
|
||||||
|
@include inset(0, 0, 0, 0);
|
||||||
|
visibility: hidden;
|
||||||
|
opacity: 0;
|
||||||
|
transition: visibility 0s linear .12s, opacity .12s ease;
|
||||||
|
background-color: var(--loading-bg-color);
|
||||||
|
display: flex;
|
||||||
|
justify-content: center;
|
||||||
|
align-items: center;
|
||||||
|
z-index: 1;
|
||||||
|
|
||||||
|
>div {
|
||||||
|
background-color: var(--loading-fore-color);
|
||||||
|
border-radius: var(--loading-border-radius);
|
||||||
|
|
||||||
|
>svg {
|
||||||
|
width: var(--loading-size);
|
||||||
|
height: var(--loading-size);
|
||||||
|
padding: 20px;
|
||||||
|
animation: loading-spinner 1.2s infinite linear;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -1,3 +1,4 @@
|
|||||||
@import './checkbox.scss';
|
@import './checkbox.scss';
|
||||||
@import './dropdown.scss';
|
@import './dropdown.scss';
|
||||||
@import './tooltip.scss';
|
@import './tooltip.scss';
|
||||||
|
@import './grid.scss';
|
@ -21,6 +21,7 @@
|
|||||||
<li data-page="lib/ui/checkbox.html">checkbox</li>
|
<li data-page="lib/ui/checkbox.html">checkbox</li>
|
||||||
<li data-page="lib/ui/tooltip.html">tooltip</li>
|
<li data-page="lib/ui/tooltip.html">tooltip</li>
|
||||||
<li data-page="lib/ui/dropdown.html">dropdown</li>
|
<li data-page="lib/ui/dropdown.html">dropdown</li>
|
||||||
|
<li data-page="lib/ui/grid.html">grid</li>
|
||||||
</ol>
|
</ol>
|
||||||
</li>
|
</li>
|
||||||
<li class="title">lib-utility</li>
|
<li class="title">lib-utility</li>
|
||||||
|
49
lib/ui/grid.html
Normal file
49
lib/ui/grid.html
Normal file
@ -0,0 +1,49 @@
|
|||||||
|
<div>
|
||||||
|
<h1>grid</h1>
|
||||||
|
<hr />
|
||||||
|
<p>
|
||||||
|
创建一个统一样式的滚动列表元素。
|
||||||
|
</p>
|
||||||
|
<hr />
|
||||||
|
<h2>示例</h2>
|
||||||
|
<pre></pre>
|
||||||
|
<div id="grid-sample"></div>
|
||||||
|
<!-- <div style="height: 80px"></div> -->
|
||||||
|
<script type="text/javascript">
|
||||||
|
!function () {
|
||||||
|
const Grid = window["lib-ui"].Grid;
|
||||||
|
|
||||||
|
const grid = new Grid(document.querySelector('#grid-sample'));
|
||||||
|
// grid.height = 0;
|
||||||
|
grid.columns = [
|
||||||
|
{ key: 'c1', caption: 'column 1' },
|
||||||
|
{ key: 'c2', caption: 'column 2', allcheck: true, type: Grid.ColumnTypes.Checkbox, enabled: 'enabled' },
|
||||||
|
{ key: 'c3', caption: 'column 3', width: 90 },
|
||||||
|
{ key: 'c4', caption: 'Note', type: Grid.ColumnTypes.Input }
|
||||||
|
];
|
||||||
|
grid.cellClicked = (rId, cId) => console.log(`row (${rId}), column (${cId}) clicked.`);
|
||||||
|
grid.init();
|
||||||
|
const source = [];
|
||||||
|
for (let i = 0; i < 1000; i++) {
|
||||||
|
source.push(
|
||||||
|
{ c1: 'abc', c2: true, c3: 12345, c4: 'Note note this is note', enabled: false },
|
||||||
|
{ c1: 'abc2bbbbaaaaa', c2: false, c3: 1225, c4: 'Note note this is note' },
|
||||||
|
{ c1: 'type', c2: false, c3: 121111 },
|
||||||
|
{ c1: 'diff', c2: true, c3: 124445555555555555 }
|
||||||
|
);
|
||||||
|
}
|
||||||
|
grid.source = source;
|
||||||
|
|
||||||
|
window.grid = grid;
|
||||||
|
}();
|
||||||
|
</script>
|
||||||
|
<style type="text/css">
|
||||||
|
#grid-sample {
|
||||||
|
height: 400px;
|
||||||
|
}
|
||||||
|
|
||||||
|
#grid-sample>.grid {
|
||||||
|
height: 100%;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
</div>
|
638
lib/ui/grid.js
638
lib/ui/grid.js
@ -21,23 +21,35 @@ class GridColumn {
|
|||||||
static setValue(element, val) { element.innerText = val }
|
static setValue(element, val) { element.innerText = val }
|
||||||
|
|
||||||
static getValue(element) { return element.innerText }
|
static getValue(element) { return element.innerText }
|
||||||
|
|
||||||
|
static setStyle(element, style) {
|
||||||
|
for (let css of Object.entries(style)) {
|
||||||
|
element.style.setProperty(css[0], css[1]);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
class GridInputColumn extends GridColumn {
|
class GridInputColumn extends GridColumn {
|
||||||
static createEdit(trigger) {
|
static createEdit(trigger) {
|
||||||
const input = document.createElement('input');
|
const input = document.createElement('input');
|
||||||
input.setAttribute('type', 'input');
|
input.setAttribute('type', 'text');
|
||||||
if (typeof trigger === 'function') {
|
if (typeof trigger === 'function') {
|
||||||
input.addEventListener('change', trigger);
|
input.addEventListener('change', trigger);
|
||||||
}
|
}
|
||||||
return input;
|
return input;
|
||||||
}
|
}
|
||||||
|
|
||||||
static setValue(element, val) { element.value = val }
|
static setValue(element, val) {
|
||||||
|
if (element.tagName !== 'INPUT') {
|
||||||
|
super.setValue(element, val);
|
||||||
|
} else {
|
||||||
|
element.value = val;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
static getValue(element) { return element.value }
|
static getValue(e) { return e.target.value }
|
||||||
|
|
||||||
static setEnabled(element, enabled) { element.disabled = enabled !== false }
|
static setEnabled(element, enabled) { element.disabled = enabled === false }
|
||||||
}
|
}
|
||||||
|
|
||||||
class GridDropdownColumn extends GridColumn {
|
class GridDropdownColumn extends GridColumn {
|
||||||
@ -53,9 +65,9 @@ class GridCheckboxColumn extends GridColumn {
|
|||||||
|
|
||||||
static setValue(element, val) { element.querySelector('input').checked = val }
|
static setValue(element, val) { element.querySelector('input').checked = val }
|
||||||
|
|
||||||
static getValue(element) { return element.querySelector('input').checked }
|
static getValue(e) { return e.target.checked }
|
||||||
|
|
||||||
static setEnabled(element, enabled) { element.querySelector('input').disabled = enabled !== false }
|
static setEnabled(element, enabled) { element.querySelector('input').disabled = enabled === false }
|
||||||
}
|
}
|
||||||
|
|
||||||
const ColumnTypes = {
|
const ColumnTypes = {
|
||||||
@ -81,6 +93,9 @@ class Grid {
|
|||||||
#rowCount = -1;
|
#rowCount = -1;
|
||||||
#overflows;
|
#overflows;
|
||||||
#scrollTop;
|
#scrollTop;
|
||||||
|
#scrollLeft;
|
||||||
|
#colTypes = {};
|
||||||
|
#colAttrs = {};
|
||||||
|
|
||||||
columns = [];
|
columns = [];
|
||||||
langs = {
|
langs = {
|
||||||
@ -89,15 +104,17 @@ class Grid {
|
|||||||
reset: r('reset', 'Reset')
|
reset: r('reset', 'Reset')
|
||||||
};
|
};
|
||||||
virtualCount = 100;
|
virtualCount = 100;
|
||||||
rowHeight = 27;
|
rowHeight = 39;
|
||||||
filterRowHeight = 26;
|
filterRowHeight = 30;
|
||||||
height;
|
height;
|
||||||
|
readonly;
|
||||||
multiSelect = false;
|
multiSelect = false;
|
||||||
|
fullrowClick = true;
|
||||||
allowHtml = false;
|
allowHtml = false;
|
||||||
holderDisabled = false;
|
holderDisabled = false;
|
||||||
window = global;
|
window = global;
|
||||||
sortIndex = -1;
|
sortIndex = -1;
|
||||||
sortDirection = 'asc';
|
sortDirection = 1;
|
||||||
|
|
||||||
willSelect;
|
willSelect;
|
||||||
selectedRowChanged;
|
selectedRowChanged;
|
||||||
@ -110,30 +127,32 @@ class Grid {
|
|||||||
Common: 0,
|
Common: 0,
|
||||||
Input: 1,
|
Input: 1,
|
||||||
Dropdown: 2,
|
Dropdown: 2,
|
||||||
Checkbox: 3
|
Checkbox: 3,
|
||||||
|
isCheckbox(type) { return type === 3 }
|
||||||
};
|
};
|
||||||
|
|
||||||
constructor(container) {
|
constructor(container) {
|
||||||
this.#parent = container;
|
this.#parent = container;
|
||||||
}
|
}
|
||||||
|
|
||||||
get source() { return this.#source; }
|
get source() { return this.#source?.map(s => s.values) }
|
||||||
set source(list) {
|
set source(list) {
|
||||||
if (this.#el == null) {
|
if (this.#el == null) {
|
||||||
throw new Error('grid has not been initialized.')
|
throw new Error('grid has not been initialized.')
|
||||||
}
|
}
|
||||||
|
if (!Array.isArray(list)) {
|
||||||
|
throw new Error('source is not an Array.')
|
||||||
|
}
|
||||||
|
list = list.map(i => { return { values: i } });
|
||||||
this.#source = list;
|
this.#source = list;
|
||||||
// TODO: filter to currentSource;
|
// TODO: filter to currentSource;
|
||||||
this.#currentSource = list;
|
this.#currentSource = list;
|
||||||
this.#containerHeight = list.length * this.rowHeight;
|
|
||||||
this.#overflows = {};
|
this.#overflows = {};
|
||||||
this.#selectedColumnIndex = -1;
|
this.#selectedColumnIndex = -1;
|
||||||
this.#selectedIndexes = [];
|
this.#selectedIndexes = [];
|
||||||
this.#startIndex = 0;
|
this.#startIndex = 0;
|
||||||
this.#scrollTop = 0;
|
this.#scrollTop = 0;
|
||||||
this.#refs.body.scrollTop = 0;
|
this.#scrollLeft = 0;
|
||||||
this.#refs.bodyContent.style.top = '0px';
|
|
||||||
this.#refs.bodyContainer.style.height = `${this.#containerHeight}px`;
|
|
||||||
this.#rowCount = -1;
|
this.#rowCount = -1;
|
||||||
|
|
||||||
if (this.sortIndex >= 0) {
|
if (this.sortIndex >= 0) {
|
||||||
@ -142,11 +161,38 @@ class Grid {
|
|||||||
this.resize();
|
this.resize();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
get virtual() { return this.#currentSource?.length > this.virtualCount }
|
get virtual() { return this.#currentSource?.length > this.virtualCount }
|
||||||
get sortKey() { }
|
|
||||||
|
get sortKey() {
|
||||||
|
if (this.columns == null) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
return this.columns[this.sortIndex]?.key;
|
||||||
|
}
|
||||||
|
|
||||||
get selectedIndexes() { return this.#selectedIndexes }
|
get selectedIndexes() { return this.#selectedIndexes }
|
||||||
set selectedIndexes(indexes) { }
|
set selectedIndexes(indexes) {
|
||||||
get selectedIndex() { }
|
const startIndex = this.#startIndex;
|
||||||
|
this.#selectedIndexes.splice(0, this.#selectedIndexes.length, ...indexes);
|
||||||
|
if (this.readonly !== true) {
|
||||||
|
this.refresh();
|
||||||
|
} else {
|
||||||
|
[...this.#refs.bodyContent.children].forEach((row, i) => {
|
||||||
|
if (indexes.indexOf(startIndex + i) >= 0) {
|
||||||
|
row.classList.add('selected');
|
||||||
|
} else if (row.classList.contains('selected')) {
|
||||||
|
row.classList.remove('selected');
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
if (typeof this.selectedRowChanged === 'function') {
|
||||||
|
this.selectedRowChanged();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
get selectedIndex() { return (this.#selectedIndexes && this.#selectedIndexes[0]) ?? -1 }
|
||||||
|
|
||||||
get loading() { return this.#refs.loading?.style.visibility === 'visible' }
|
get loading() { return this.#refs.loading?.style.visibility === 'visible' }
|
||||||
set loading(flag) {
|
set loading(flag) {
|
||||||
if (this.#refs.loading == null) {
|
if (this.#refs.loading == null) {
|
||||||
@ -160,6 +206,7 @@ class Grid {
|
|||||||
this.#refs.loading.style.opacity = 1;
|
this.#refs.loading.style.opacity = 1;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
get scrollTop() { return this.#refs.body?.scrollTop; }
|
get scrollTop() { return this.#refs.body?.scrollTop; }
|
||||||
set scrollTop(top) {
|
set scrollTop(top) {
|
||||||
if (this.#refs.body == null) {
|
if (this.#refs.body == null) {
|
||||||
@ -173,7 +220,6 @@ class Grid {
|
|||||||
this.#el = null;
|
this.#el = null;
|
||||||
this.#refs = {};
|
this.#refs = {};
|
||||||
this.#rendering = true;
|
this.#rendering = true;
|
||||||
this.#currentSource = this.source;
|
|
||||||
if (!(container instanceof HTMLElement)) {
|
if (!(container instanceof HTMLElement)) {
|
||||||
throw new Error('no specified parent.');
|
throw new Error('no specified parent.');
|
||||||
}
|
}
|
||||||
@ -186,26 +232,21 @@ class Grid {
|
|||||||
let flag = false;
|
let flag = false;
|
||||||
if (e.key === 'ArrowUp') {
|
if (e.key === 'ArrowUp') {
|
||||||
// up
|
// up
|
||||||
flag = true;
|
if (index > 0) {
|
||||||
if (index > 1) {
|
flag = true;
|
||||||
delete this.#currentSource[index].__selected;
|
|
||||||
index -= 1;
|
index -= 1;
|
||||||
} else {
|
|
||||||
index = 0;
|
|
||||||
}
|
}
|
||||||
} else if (e.key === 'ArrowDown') {
|
} else if (e.key === 'ArrowDown') {
|
||||||
// down
|
// down
|
||||||
flag = true;
|
|
||||||
const count = this.#currentSource?.length ?? 0;
|
const count = this.#currentSource?.length ?? 0;
|
||||||
if (index < count - 1) {
|
if (index < count - 1) {
|
||||||
delete this.#currentSource[index].__selected;
|
flag = true;
|
||||||
index += 1;
|
index += 1;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (flag) {
|
if (flag) {
|
||||||
this.selectedIndexes = [index];
|
this.#selectedIndexes = [index];
|
||||||
this.scrollToIndex(index);
|
this.scrollToIndex(index);
|
||||||
this.#currentSource[index].__selected = true;
|
|
||||||
this.refresh();
|
this.refresh();
|
||||||
if (typeof this.selectedRowChanged === 'function') {
|
if (typeof this.selectedRowChanged === 'function') {
|
||||||
this.selectedRowChanged(index);
|
this.selectedRowChanged(index);
|
||||||
@ -228,21 +269,22 @@ class Grid {
|
|||||||
// loading
|
// loading
|
||||||
const loading = document.createElement('div');
|
const loading = document.createElement('div');
|
||||||
loading.className = 'grid-loading';
|
loading.className = 'grid-loading';
|
||||||
loading.appendChild(createIcon('fa-regular', 'spinner-third'));
|
const loadingHolder = document.createElement('div');
|
||||||
|
loadingHolder.appendChild(createIcon('fa-regular', 'spinner-third'));
|
||||||
|
loading.appendChild(loadingHolder);
|
||||||
this.#refs.loading = loading;
|
this.#refs.loading = loading;
|
||||||
grid.appendChild(loading);
|
grid.appendChild(loading);
|
||||||
this.#el = grid;
|
this.#el = grid;
|
||||||
|
|
||||||
this.#rendering = false;
|
this.#rendering = false;
|
||||||
if (this.sortIndex >= 0) {
|
if (this.sortIndex >= 0) {
|
||||||
this.sortColumn(true);
|
this.sortColumn();
|
||||||
} else {
|
|
||||||
this.resize();
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
scrollToIndex(index) {
|
scrollToIndex(index) {
|
||||||
this.#scrollToTop(index * this.rowHeight, true);
|
const top = this.#scrollToTop(index * this.rowHeight, true);
|
||||||
|
this.#refs.body.scrollTop = top;
|
||||||
}
|
}
|
||||||
|
|
||||||
resize(force) {
|
resize(force) {
|
||||||
@ -250,21 +292,18 @@ class Grid {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
const body = this.#refs.body;
|
const body = this.#refs.body;
|
||||||
let height = this.#refs.header.offsetHeight + 2;
|
// let height = this.#refs.header.offsetHeight + 2;
|
||||||
let top = body.offsetTop;
|
// let top = body.offsetTop;
|
||||||
if (top !== height) {
|
// if (top !== height) {
|
||||||
body.style.top = `${height}px`;
|
// body.style.top = `${height}px`;
|
||||||
top = height;
|
// top = height;
|
||||||
}
|
// }
|
||||||
|
const top = this.#refs.header.offsetHeight;
|
||||||
|
|
||||||
height = this.height;
|
let height = this.height;
|
||||||
if (isNaN(height)) {
|
if (isNaN(height) || height <= 0) {
|
||||||
height = this.#el.offsetHeight - top;
|
height = this.#el.offsetHeight - top;
|
||||||
} else if (height === 0) {
|
|
||||||
height = this.#refs.bodyContent.offsetHeight;
|
|
||||||
this.#el.style.height = `${top + height}px`;
|
|
||||||
}
|
}
|
||||||
body.style.height = `${height}px`;
|
|
||||||
const count = truncate((height - 1) / this.rowHeight) * (RedumCount * 2) + 1;
|
const count = truncate((height - 1) / this.rowHeight) * (RedumCount * 2) + 1;
|
||||||
if (force || count !== this.#rowCount) {
|
if (force || count !== this.#rowCount) {
|
||||||
this.#rowCount = count;
|
this.#rowCount = count;
|
||||||
@ -275,6 +314,10 @@ class Grid {
|
|||||||
|
|
||||||
reload() {
|
reload() {
|
||||||
this.#containerHeight = this.#currentSource.length * this.rowHeight;
|
this.#containerHeight = this.#currentSource.length * this.rowHeight;
|
||||||
|
this.#refs.body.scrollTop = 0;
|
||||||
|
this.#refs.body.scrollLeft = 0;
|
||||||
|
this.#refs.bodyContent.style.top = '0px';
|
||||||
|
this.#refs.bodyContainer.style.height = `${this.#containerHeight}px`;
|
||||||
this.#adjustRows(this.#refs.bodyContent);
|
this.#adjustRows(this.#refs.bodyContent);
|
||||||
this.refresh();
|
this.refresh();
|
||||||
}
|
}
|
||||||
@ -292,7 +335,10 @@ class Grid {
|
|||||||
if (!col.autoResize) {
|
if (!col.autoResize) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
const width = widths[i];
|
let width = widths[i];
|
||||||
|
if (width < col.width) {
|
||||||
|
width = col.width;
|
||||||
|
}
|
||||||
if (width > 0) {
|
if (width > 0) {
|
||||||
this.#changeColumnWidth(i, width);
|
this.#changeColumnWidth(i, width);
|
||||||
}
|
}
|
||||||
@ -300,7 +346,79 @@ class Grid {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
sortColumn(auto, reload) { }
|
resetChange() {
|
||||||
|
if (this.#currentSource == null) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
for (let row of this.#currentSource) {
|
||||||
|
delete row.__changed;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
sortColumn(reload) {
|
||||||
|
const index = this.sortIndex;
|
||||||
|
const col = this.columns[index];
|
||||||
|
if (col == null) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const direction = this.sortDirection;
|
||||||
|
[...this.#refs.header.children].forEach((th, i) => {
|
||||||
|
const arrow = th.children[1]; // th.querySelector('layer.arrow');
|
||||||
|
if (arrow == null) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (i === index) {
|
||||||
|
arrow.className = `arrow ${(direction !== 1 ? 'desc' : 'asc')}`;
|
||||||
|
} else if (arrow.className !== 'arrow') {
|
||||||
|
arrow.className = 'arrow';
|
||||||
|
}
|
||||||
|
});
|
||||||
|
let comparer;
|
||||||
|
if (typeof col.sortFilter !== 'function') {
|
||||||
|
const direction = this.sortDirection;
|
||||||
|
if (isNaN(direction)) {
|
||||||
|
direction = 1;
|
||||||
|
}
|
||||||
|
comparer = (a, b) => {
|
||||||
|
const ta = a.values[col.key];
|
||||||
|
const tb = b.values[col.key];
|
||||||
|
if ((ta == null || tb == null) && typeof col.filter === 'function') {
|
||||||
|
a = col.filter(a.values);
|
||||||
|
b = col.filter(b.values);
|
||||||
|
} else {
|
||||||
|
a = ta;
|
||||||
|
b = tb;
|
||||||
|
}
|
||||||
|
if (a?.value != null) {
|
||||||
|
a = a.value;
|
||||||
|
}
|
||||||
|
if (b?.value != null) {
|
||||||
|
b = b.value;
|
||||||
|
}
|
||||||
|
if (a == null && typeof b === 'number') {
|
||||||
|
a = 0;
|
||||||
|
} else if (typeof a === 'number' && b == null) {
|
||||||
|
b = 0;
|
||||||
|
} else if (a != null && b == null) {
|
||||||
|
return direction;
|
||||||
|
} else if (typeof a === 'string' && typeof b === 'string') {
|
||||||
|
a = a.toLowerCase();
|
||||||
|
b = b.toLowerCase();
|
||||||
|
}
|
||||||
|
return a === b ? 0 : (a > b ? 1 : -1) * direction;
|
||||||
|
};
|
||||||
|
} else {
|
||||||
|
comparer = (a, b) => col.sortFilter(a, b) * direction;
|
||||||
|
}
|
||||||
|
this.#source.sort(comparer);
|
||||||
|
// TODO: filter to currentSource;
|
||||||
|
this.#currentSource = this.#source;
|
||||||
|
if (reload) {
|
||||||
|
this.reload();
|
||||||
|
} else {
|
||||||
|
this.refresh();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
#createHeader() {
|
#createHeader() {
|
||||||
const thead = document.createElement('table');
|
const thead = document.createElement('table');
|
||||||
@ -320,29 +438,34 @@ class Grid {
|
|||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
// style
|
// style
|
||||||
|
const isCheckbox = Grid.ColumnTypes.isCheckbox(col.type);
|
||||||
if (col.width > 0 || col.shrink) {
|
if (col.width > 0 || col.shrink) {
|
||||||
col.autoResize = false;
|
col.autoResize = false;
|
||||||
} else {
|
} else {
|
||||||
col.autoResize = true;
|
col.autoResize = true;
|
||||||
this.#needResize = true;
|
this.#needResize = true;
|
||||||
sizer.innerText = col.caption;
|
sizer.innerText = col.caption;
|
||||||
let width = sizer.offsetWidth + 20;
|
let width = sizer.offsetWidth + 22;
|
||||||
|
if (col.allcheck && isCheckbox) {
|
||||||
|
width += 32;
|
||||||
|
}
|
||||||
if (width < MiniColumnWidth) {
|
if (width < MiniColumnWidth) {
|
||||||
width = MiniColumnWidth;
|
width = MiniColumnWidth;
|
||||||
}
|
}
|
||||||
col.width = width;
|
col.width = width;
|
||||||
}
|
}
|
||||||
col.align ??= 'left';
|
col.align ??= isCheckbox ? 'center' : 'left';
|
||||||
if (col.sortable !== false) {
|
if (col.sortable !== false) {
|
||||||
col.sortable = true;
|
col.sortable = true;
|
||||||
}
|
}
|
||||||
if (col.shrink) {
|
if (col.shrink) {
|
||||||
col.style = { 'text-align': col.align };
|
col.style = { 'text-align': col.align };
|
||||||
} else {
|
} else {
|
||||||
|
const w = `${col.width}px`;
|
||||||
col.style = {
|
col.style = {
|
||||||
'width': col.width,
|
'width': w,
|
||||||
'max-width': col.width,
|
'max-width': w,
|
||||||
'min-width': col.width,
|
'min-width': w,
|
||||||
'text-align': col.align
|
'text-align': col.align
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
@ -357,14 +480,19 @@ class Grid {
|
|||||||
th.addEventListener('mousedown', e => this.#onDragStart(col, e));
|
th.addEventListener('mousedown', e => this.#onDragStart(col, e));
|
||||||
const wrapper = document.createElement('div');
|
const wrapper = document.createElement('div');
|
||||||
th.appendChild(wrapper);
|
th.appendChild(wrapper);
|
||||||
if (col.enabled !== false && col.allcheck && col.type === Grid.ColumnTypes.Checkbox) {
|
if (col.enabled !== false && col.allcheck && isCheckbox) {
|
||||||
const check = createCheckbox({
|
const check = createCheckbox({
|
||||||
onchange: e => this.#onColumnAllChecked(col, e.target.checked)
|
onchange: e => this.#onColumnAllChecked(col, e.target.checked)
|
||||||
});
|
});
|
||||||
wrapper.appendChild(check);
|
wrapper.appendChild(check);
|
||||||
}
|
}
|
||||||
const caption = document.createElement('span');
|
const caption = document.createElement('span');
|
||||||
caption = col.caption;
|
if (col.textStyle != null) {
|
||||||
|
for (let css of Object.entries(col.textStyle)) {
|
||||||
|
caption.style.setProperty(css[0], css[1]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
caption.innerText = col.caption;
|
||||||
wrapper.appendChild(caption);
|
wrapper.appendChild(caption);
|
||||||
// order arrow
|
// order arrow
|
||||||
if (col.sortable) {
|
if (col.sortable) {
|
||||||
@ -407,25 +535,17 @@ class Grid {
|
|||||||
body.className = 'grid-body';
|
body.className = 'grid-body';
|
||||||
body.addEventListener('scroll', e => throttle(this.#onScroll, RefreshInterval, this, e), { passive: true });
|
body.addEventListener('scroll', e => throttle(this.#onScroll, RefreshInterval, this, e), { passive: true });
|
||||||
const cols = this.columns;
|
const cols = this.columns;
|
||||||
let height = this.#currentSource.length * this.rowHeight;
|
let width = 1;
|
||||||
let width;
|
for (let col of cols) {
|
||||||
if (height === 0) {
|
if (col.visible !== false && !isNaN(col.width)) {
|
||||||
height = 1;
|
width += col.width + 1;
|
||||||
width = 0;
|
|
||||||
for (let col of cols) {
|
|
||||||
if (col.visible !== false && !isNaN(col.width)) {
|
|
||||||
width += col.width + 1;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
width += 1;
|
|
||||||
}
|
}
|
||||||
this.#containerHeight = height;
|
|
||||||
// body container
|
// body container
|
||||||
const bodyContainer = document.createElement('div');
|
const bodyContainer = document.createElement('div');
|
||||||
bodyContainer.style.position = 'relative';
|
bodyContainer.style.position = 'relative';
|
||||||
bodyContainer.style.minWidth = '100%';
|
bodyContainer.style.minWidth = '100%';
|
||||||
bodyContainer.style.minHeight = '1px';
|
bodyContainer.style.minHeight = '1px';
|
||||||
bodyContainer.style.height = `${height}px`;
|
|
||||||
if (width > 0) {
|
if (width > 0) {
|
||||||
bodyContainer.style.width = `${width}px`;
|
bodyContainer.style.width = `${width}px`;
|
||||||
}
|
}
|
||||||
@ -434,7 +554,7 @@ class Grid {
|
|||||||
const bodyContent = document.createElement('table');
|
const bodyContent = document.createElement('table');
|
||||||
bodyContent.className = 'grid-body-content';
|
bodyContent.className = 'grid-body-content';
|
||||||
bodyContainer.appendChild(bodyContent);
|
bodyContainer.appendChild(bodyContent);
|
||||||
this.#adjustRows();
|
// this.#adjustRows();
|
||||||
// events
|
// events
|
||||||
if (!this.holderDisabled) {
|
if (!this.holderDisabled) {
|
||||||
const holder = document.createElement('div');
|
const holder = document.createElement('div');
|
||||||
@ -464,7 +584,7 @@ class Grid {
|
|||||||
for (let i = 0; i < count; i += 1) {
|
for (let i = 0; i < count; i += 1) {
|
||||||
const row = document.createElement('tr');
|
const row = document.createElement('tr');
|
||||||
row.className = 'grid-row';
|
row.className = 'grid-row';
|
||||||
row.addEventListener('mousedown', e => this.#onRowClicked(e, exists + 1));
|
row.addEventListener('mousedown', e => this.#onRowClicked(e, exists + i));
|
||||||
row.addEventListener('dblclick', e => this.#onRowDblClicked(e));
|
row.addEventListener('dblclick', e => this.#onRowDblClicked(e));
|
||||||
cols.forEach((col, j) => {
|
cols.forEach((col, j) => {
|
||||||
const cell = document.createElement('td');
|
const cell = document.createElement('td');
|
||||||
@ -480,13 +600,25 @@ class Grid {
|
|||||||
cell.style.setProperty(css[0], css[1]);
|
cell.style.setProperty(css[0], css[1]);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (col.type === Grid.ColumnTypes.Checkbox) {
|
if (Grid.ColumnTypes.isCheckbox(col.type)) {
|
||||||
cell.appendChild(GridCheckboxColumn.createEdit(e => this.#onRowChanged(e, exists + i, col, e.target.checked)));
|
cell.appendChild(GridCheckboxColumn.createEdit(e => this.#onRowChanged(e, exists + i, col, e.target.checked)));
|
||||||
} else if (this.allowHtml && col.type != null && isNaN(col.type)) {
|
// this.#colTypes[col.key] = GridCheckboxColumn;
|
||||||
cell.appendChild(col.type.create());
|
|
||||||
} else {
|
} else {
|
||||||
cell.appendChild(GridColumn.create());
|
let type = this.#colTypes[col.key];
|
||||||
|
if (type == null) {
|
||||||
|
if (isNaN(col.type)) {
|
||||||
|
if (this.allowHtml && col.type != null) {
|
||||||
|
type = col.type;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
type = ColumnTypes[col.type];
|
||||||
|
}
|
||||||
|
type ??= GridColumn;
|
||||||
|
this.#colTypes[col.key] = type;
|
||||||
|
}
|
||||||
|
cell.appendChild(type.create());
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
row.appendChild(cell);
|
row.appendChild(cell);
|
||||||
});
|
});
|
||||||
@ -502,10 +634,9 @@ class Grid {
|
|||||||
}
|
}
|
||||||
|
|
||||||
#fillRows(rows, cols, widths) {
|
#fillRows(rows, cols, widths) {
|
||||||
const startIndex = this.startIndex;
|
const startIndex = this.#startIndex;
|
||||||
const selected = this.#selectedIndexes;
|
const selectedIndexes = this.#selectedIndexes;
|
||||||
const allowHtml = this.allowHtml;
|
[...rows].forEach((row, i) => {
|
||||||
rows.forEach((row, i) => {
|
|
||||||
const vals = this.#currentSource[startIndex + i];
|
const vals = this.#currentSource[startIndex + i];
|
||||||
if (vals == null) {
|
if (vals == null) {
|
||||||
return;
|
return;
|
||||||
@ -514,13 +645,19 @@ class Grid {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
const item = vals.values;
|
const item = vals.values;
|
||||||
if (selected.indexOf(startIndex + i) < 0) {
|
const selected = selectedIndexes.indexOf(startIndex + i) >= 0;
|
||||||
row.classList.remove('selected');
|
if (selected) {
|
||||||
} else {
|
|
||||||
row.classList.add('selected');
|
row.classList.add('selected');
|
||||||
|
} else if (row.classList.contains('selected')) {
|
||||||
|
row.classList.remove('selected');
|
||||||
}
|
}
|
||||||
// data
|
// data
|
||||||
const selected = row.dataset.selected === '1';
|
const selectChanged = vals.__selected ^ selected;
|
||||||
|
if (selected) {
|
||||||
|
vals.__selected = true;
|
||||||
|
} else {
|
||||||
|
delete vals.__selected;
|
||||||
|
}
|
||||||
cols.forEach((col, j) => {
|
cols.forEach((col, j) => {
|
||||||
if (col.visible === false) {
|
if (col.visible === false) {
|
||||||
return;
|
return;
|
||||||
@ -539,19 +676,18 @@ class Grid {
|
|||||||
val ??= '';
|
val ??= '';
|
||||||
// fill
|
// fill
|
||||||
const cell = row.children[j];
|
const cell = row.children[j];
|
||||||
const custom = allowHtml && col.type != null && isNaN(col.type);
|
if (typeof col.bgFilter === 'function') {
|
||||||
|
const bgColor = col.bgFilter(item);
|
||||||
|
cell.style.backgroundColor = bgColor ?? '';
|
||||||
|
}
|
||||||
|
const isCheckbox = Grid.ColumnTypes.isCheckbox(col.type);
|
||||||
|
const type = isCheckbox ? GridCheckboxColumn : this.#colTypes[col.key] ?? GridColumn;
|
||||||
let element;
|
let element;
|
||||||
if (vals.__selected ^ selected) {
|
if (!isCheckbox && selectChanged) {
|
||||||
if (custom) {
|
element = selected && typeof type.createEdit === 'function' ?
|
||||||
element = selected ?
|
type.createEdit(e => this.#onRowChanged(e, startIndex + i, col, type.getValue(e))) :
|
||||||
col.type.createEdit(e => this.#onRowChanged(e, startIndex + i, col, col.type.getValue(element))) :
|
type.create();
|
||||||
col.type.create();
|
cell.replaceChildren(element);
|
||||||
cell.replaceChildren(element);
|
|
||||||
// } else if (col.type !== Grid.ColumnTypes.Checkbox) {
|
|
||||||
// // TODO:
|
|
||||||
} else {
|
|
||||||
element = cell.children[0];
|
|
||||||
}
|
|
||||||
} else {
|
} else {
|
||||||
element = cell.children[0];
|
element = cell.children[0];
|
||||||
}
|
}
|
||||||
@ -559,44 +695,322 @@ class Grid {
|
|||||||
if (typeof enabled === 'string') {
|
if (typeof enabled === 'string') {
|
||||||
enabled = item[enabled];
|
enabled = item[enabled];
|
||||||
}
|
}
|
||||||
if (custom) {
|
type.setValue(element, val, item);
|
||||||
col.type.setValue(element, item);
|
if (typeof type.setEnabled === 'function') {
|
||||||
col.type.setEnabled(element, enabled);
|
type.setEnabled(element, enabled);
|
||||||
} else if (col.type === Grid.ColumnTypes.Checkbox) {
|
|
||||||
GridCheckboxColumn.setValue(element, val);
|
|
||||||
GridCheckboxColumn.setEnabled(element, enabled);
|
|
||||||
} else {
|
|
||||||
// TODO: input, dropdown, etc...
|
|
||||||
GridColumn.setValue(element, val);
|
|
||||||
}
|
}
|
||||||
})
|
// auto resize
|
||||||
|
if (this.#needResize && col.autoResize) {
|
||||||
|
const width = cell.scrollWidth + 12;
|
||||||
|
if (width > 0 && widths != null && (isNaN(widths[j]) || widths[j] < width)) {
|
||||||
|
widths[j] = width;
|
||||||
|
widths.flag = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (typeof col.styleFilter === 'function') {
|
||||||
|
const style = col.styleFilter(item);
|
||||||
|
if (style != null) {
|
||||||
|
type.setStyle(element, style);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (col.events != null) {
|
||||||
|
for (let ev of Object.entries(col.events)) {
|
||||||
|
element[ev[0]] = ev[1].bind(item);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (col.attrs != null) {
|
||||||
|
let attrs = col.attrs;
|
||||||
|
if (typeof attrs === 'function') {
|
||||||
|
attrs = attrs(item);
|
||||||
|
}
|
||||||
|
for (let attr of Object.entries(attrs)) {
|
||||||
|
element.setAttribute(attr[0], attr[1]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
#changeColumnWidth(i, width) { }
|
#changeColumnWidth(index, width) {
|
||||||
|
const col = this.columns[index];
|
||||||
|
// const oldwidth = col.width;
|
||||||
|
const w = `${width}px`;
|
||||||
|
col.width = width;
|
||||||
|
col.style.width = w;
|
||||||
|
col.style['max-width'] = w;
|
||||||
|
col.style['min-width'] = w;
|
||||||
|
let element = this.#refs.header.children[index];
|
||||||
|
element.style.width = w;
|
||||||
|
element.style.maxWidth = w;
|
||||||
|
element.style.minWidth = w;
|
||||||
|
const body = this.#refs.bodyContent;
|
||||||
|
for (let row of body.children) {
|
||||||
|
element = row.children[index];
|
||||||
|
if (element != null) {
|
||||||
|
element.style.width = w;
|
||||||
|
element.style.maxWidth = w;
|
||||||
|
element.style.minWidth = w;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// } else {
|
||||||
|
// width = this.#refs.bodyContainer.offsetWidth - oldwidth + width;
|
||||||
|
// this.#refs.bodyContainer.style.width = `${width}px`;
|
||||||
|
// }
|
||||||
|
}
|
||||||
|
|
||||||
#scrollToTop(top, reload) { }
|
#scrollToTop(top, reload) {
|
||||||
|
const rowHeight = this.rowHeight;
|
||||||
|
top -= (top % (rowHeight * 2)) + (RedumCount * rowHeight);
|
||||||
|
if (top < 0) {
|
||||||
|
top = 0;
|
||||||
|
} else {
|
||||||
|
let bottomTop = this.#containerHeight - (reload ? 0 : this.#rowCount * rowHeight);
|
||||||
|
if (bottomTop < 0) {
|
||||||
|
bottomTop = 0;
|
||||||
|
}
|
||||||
|
if (top > bottomTop) {
|
||||||
|
top = bottomTop;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (this.#scrollTop !== top) {
|
||||||
|
this.#scrollTop = top;
|
||||||
|
if (this.virtual) {
|
||||||
|
this.#startIndex = top / rowHeight;
|
||||||
|
}
|
||||||
|
this.refresh();
|
||||||
|
if (this.virtual) {
|
||||||
|
this.#refs.bodyContent.style.top = `${top}px`;
|
||||||
|
}
|
||||||
|
} else if (reload) {
|
||||||
|
this.refresh();
|
||||||
|
}
|
||||||
|
|
||||||
|
return top;
|
||||||
|
}
|
||||||
|
|
||||||
|
#getColumnIndex(target) {
|
||||||
|
if (target == null) {
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
let parent;
|
||||||
|
while ((parent = target.parentElement) != null && !parent.classList.contains('grid-row')) {
|
||||||
|
target = parent;
|
||||||
|
}
|
||||||
|
if (parent == null) {
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
const index = [...parent.children].indexOf(target);
|
||||||
|
return index >= this.columns.length ? -1 : index;
|
||||||
|
}
|
||||||
|
|
||||||
|
#onHeaderClicked(col, e, force) {
|
||||||
|
const attr = this.#colAttrs[col.key];
|
||||||
|
if (!force && attr != null && (attr.resizing || attr.dragging)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (col.sortable && ['LABEL', 'LAYER', 'SVG', 'USE'].indexOf(e.target.tagName) < 0) {
|
||||||
|
const index = this.columns.indexOf(col);
|
||||||
|
if (index < 0) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (this.sortIndex === index) {
|
||||||
|
this.sortDirection = this.sortDirection === 1 ? -1 : 1;
|
||||||
|
} else {
|
||||||
|
this.sortIndex = index;
|
||||||
|
}
|
||||||
|
this.sortColumn(true);
|
||||||
|
if (typeof this.columnChanged === 'function') {
|
||||||
|
this.columnChanged(ColumnChangedType.Sort, index, this.sortDirection);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
#onHeaderClicked(col, e, force) { }
|
|
||||||
#onDragStart(col, e) { }
|
#onDragStart(col, e) { }
|
||||||
|
|
||||||
#onResizeStart(col, e) { }
|
#onResizeStart(col, e) { }
|
||||||
#onColumnAllChecked(col, flag) { }
|
|
||||||
#onScroll(e) { }
|
#onColumnAllChecked(col, flag) {
|
||||||
#onBodyMouseMove(e, holder) { }
|
if (this.#currentSource == null) {
|
||||||
#onRowClicked(e, index, colIndex) { }
|
return;
|
||||||
#onRowDblClicked(e) { }
|
}
|
||||||
|
const key = col.key;
|
||||||
|
const test = typeof col.enabled === 'string';
|
||||||
|
if (typeof col.onallchecked === 'function') {
|
||||||
|
col.onallchecked.call(this, col, flag);
|
||||||
|
} else {
|
||||||
|
for (let row of this.#currentSource) {
|
||||||
|
const item = row.values;
|
||||||
|
if (item == null) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
const enabled = test ? item[col.enabled] : col.enabled;
|
||||||
|
if (enabled !== false) {
|
||||||
|
item[key] = flag;
|
||||||
|
row.__changed = true;
|
||||||
|
if (typeof col.onchanged === 'function') {
|
||||||
|
col.onchanged.call(this, item, flag);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
this.refresh();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#onScroll(e) {
|
||||||
|
this.#scrollLeft = e.target.scrollLeft;
|
||||||
|
if (!this.virtual) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const top = e.target.scrollTop;
|
||||||
|
this.#scrollToTop(top);
|
||||||
|
}
|
||||||
|
|
||||||
|
#onBodyMouseMove(e, holder) {
|
||||||
|
let target = e.target;
|
||||||
|
if (target.className === 'grid-hover-holder') {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
let parent;
|
||||||
|
while ((parent = target.parentElement) != null && !parent.classList.contains('grid-row')) {
|
||||||
|
target = parent;
|
||||||
|
}
|
||||||
|
let keyid = target.keyid;
|
||||||
|
if (parent == null || keyid == null) {
|
||||||
|
delete holder.keyid;
|
||||||
|
if (holder.style.display !== 'none') {
|
||||||
|
holder.style.display = 'none';
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const oldkeyid = holder.keyid;
|
||||||
|
keyid += this.#startIndex << MaxColumnBit;
|
||||||
|
if (keyid === oldkeyid) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
let overflow = this.#overflows[keyid];
|
||||||
|
if (overflow == null) {
|
||||||
|
overflow = target.scrollWidth > target.offsetWidth;
|
||||||
|
this.#overflows[keyid] = overflow;
|
||||||
|
}
|
||||||
|
if (overflow) {
|
||||||
|
holder.keyid = keyid;
|
||||||
|
holder.innerText = target.innerText;
|
||||||
|
const top = this.#refs.bodyContent.offsetTop + target.offsetTop + 1;
|
||||||
|
let left = target.offsetLeft;
|
||||||
|
let width = holder.offsetWidth;
|
||||||
|
if (width > this.#bodyClientWidth) {
|
||||||
|
width = this.#bodyClientWidth;
|
||||||
|
}
|
||||||
|
const maxleft = this.#bodyClientWidth + this.#scrollLeft - width;
|
||||||
|
if (left > maxleft) {
|
||||||
|
left = maxleft;
|
||||||
|
}
|
||||||
|
const height = target.offsetHeight;
|
||||||
|
holder.style.cssText = `top: ${top}px; left: ${left}px; max-width: ${this.#bodyClientWidth}px; height: ${height - 2}px`;
|
||||||
|
} else {
|
||||||
|
if (oldkeyid != null) {
|
||||||
|
delete holder.keyid;
|
||||||
|
}
|
||||||
|
if (holder.style.display !== 'none') {
|
||||||
|
holder.style.display = 'none';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#onRowClicked(e, index, colIndex) {
|
||||||
|
const startIndex = this.#startIndex;
|
||||||
|
const selectedIndex = startIndex + index;
|
||||||
|
if (typeof this.willSelect === 'function' && !this.willSelect(selectedIndex, colIndex)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
// multi-select
|
||||||
|
let flag = false;
|
||||||
|
const selectedIndexes = this.#selectedIndexes;
|
||||||
|
if (this.multiSelect) {
|
||||||
|
if (e.ctrlKey) {
|
||||||
|
const i = selectedIndexes.indexOf(selectedIndex);
|
||||||
|
if (i < 0) {
|
||||||
|
selectedIndexes.push(selectedIndex);
|
||||||
|
} else {
|
||||||
|
selectedIndexes.splice(i, 1);
|
||||||
|
}
|
||||||
|
flag = true;
|
||||||
|
} else if (e.shiftKey && selectedIndexes.length > 0) {
|
||||||
|
if (selectedIndexes.length > 1 || selectedIndexes[0] !== selectedIndex) {
|
||||||
|
let start = selectedIndexes[selectedIndexes.length - 1];
|
||||||
|
let end;
|
||||||
|
if (start > selectedIndex) {
|
||||||
|
end = start;
|
||||||
|
start = selectedIndex;
|
||||||
|
} else {
|
||||||
|
end = selectedIndex;
|
||||||
|
}
|
||||||
|
selectedIndexes.splice(0);
|
||||||
|
for (let i = start; i <= end; i += 1) {
|
||||||
|
selectedIndexes.push(i);
|
||||||
|
}
|
||||||
|
flag = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (!flag && selectedIndexes.length !== 1 || selectedIndexes[0] !== selectedIndex) {
|
||||||
|
selectedIndexes.splice(0, selectedIndexes.length, selectedIndex);
|
||||||
|
flag = true;
|
||||||
|
}
|
||||||
|
// apply style
|
||||||
|
if (flag) {
|
||||||
|
if (this.readonly !== true) {
|
||||||
|
this.refresh();
|
||||||
|
} else {
|
||||||
|
[...this.#refs.bodyContent.children].forEach((row, i) => {
|
||||||
|
if (selectedIndexes.indexOf(startIndex + i) >= 0) {
|
||||||
|
row.classList.add('selected');
|
||||||
|
} else if (row.classList.contains('selected')) {
|
||||||
|
row.classList.remove('selected');
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
if (typeof this.selectedRowChanged === 'function') {
|
||||||
|
this.selectedRowChanged(selectedIndex);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
colIndex ??= this.#getColumnIndex(e.target);
|
||||||
|
this.#selectedColumnIndex = colIndex;
|
||||||
|
if ((this.fullrowClick || colIndex >= 0) && e.buttons === 1 && typeof this.cellClicked === 'function') {
|
||||||
|
if (this.cellClicked(selectedIndex, colIndex) === false) {
|
||||||
|
e.stopPropagation();
|
||||||
|
e.preventDefault();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#onRowDblClicked(e) {
|
||||||
|
if (e.target.tagName === 'INPUT') {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const index = this.selectedIndex;
|
||||||
|
if (typeof this.rowDblClicked === 'function') {
|
||||||
|
this.rowDblClicked(index);
|
||||||
|
}
|
||||||
|
if (typeof this.cellDblClicked === 'function') {
|
||||||
|
const colIndex = this.#selectedColumnIndex;
|
||||||
|
if ((this.fullrowClick || colIndex >= 0) && e.buttons === 1) {
|
||||||
|
this.cellDblClicked(index, colIndex);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
#onRowChanged(_e, index, col, value) {
|
#onRowChanged(_e, index, col, value) {
|
||||||
if (this.#currentSource == null) {
|
if (this.#currentSource == null) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
const item = this.#currentSource[this.#startIndex + index].values;
|
const row = this.#currentSource[this.#startIndex + index];
|
||||||
|
const item = row.values;
|
||||||
if (item == null) {
|
if (item == null) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
const enabled = typeof col.enabled === 'string' ? item[col.enabled] : col.enabled;
|
const enabled = typeof col.enabled === 'string' ? item[col.enabled] : col.enabled;
|
||||||
if (enabled !== false) {
|
if (enabled !== false) {
|
||||||
item[col.key] = value;
|
item[col.key] = value;
|
||||||
item.__changed = true;
|
row.__changed = true;
|
||||||
if (typeof col.onchanged === 'function') {
|
if (typeof col.onchanged === 'function') {
|
||||||
col.onchanged.call(this, item, value);
|
col.onchanged.call(this, item, value);
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user