@digital-realty/ix-grid 1.0.6
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/LICENSE +21 -0
- package/README.md +62 -0
- package/demo/columns.js +65 -0
- package/demo/contacts.js +292 -0
- package/demo/index.html +92 -0
- package/dist/IxGrid.d.ts +46 -0
- package/dist/IxGrid.js +227 -0
- package/dist/IxGrid.js.map +1 -0
- package/dist/components/IxGridColumnFilter.d.ts +20 -0
- package/dist/components/IxGridColumnFilter.js +133 -0
- package/dist/components/IxGridColumnFilter.js.map +1 -0
- package/dist/components/IxGridRowFilter.d.ts +37 -0
- package/dist/components/IxGridRowFilter.js +317 -0
- package/dist/components/IxGridRowFilter.js.map +1 -0
- package/dist/components/IxPagination.d.ts +13 -0
- package/dist/components/IxPagination.js +93 -0
- package/dist/components/IxPagination.js.map +1 -0
- package/dist/components/grid-column-filter-styles.d.ts +1 -0
- package/dist/components/grid-column-filter-styles.js +71 -0
- package/dist/components/grid-column-filter-styles.js.map +1 -0
- package/dist/components/grid-row-filter-styles.d.ts +1 -0
- package/dist/components/grid-row-filter-styles.js +414 -0
- package/dist/components/grid-row-filter-styles.js.map +1 -0
- package/dist/components/pagination-styles.d.ts +1 -0
- package/dist/components/pagination-styles.js +33 -0
- package/dist/components/pagination-styles.js.map +1 -0
- package/dist/grid-view-styles.d.ts +1 -0
- package/dist/grid-view-styles.js +265 -0
- package/dist/grid-view-styles.js.map +1 -0
- package/dist/index.d.ts +1 -0
- package/dist/index.js +2 -0
- package/dist/index.js.map +1 -0
- package/dist/ix-grid-copy.d.ts +10 -0
- package/dist/ix-grid-copy.js +11 -0
- package/dist/ix-grid-copy.js.map +1 -0
- package/dist/ix-grid.d.ts +1 -0
- package/dist/ix-grid.js +3 -0
- package/dist/ix-grid.js.map +1 -0
- package/dist/test/ix-grid-column-filter.test.d.ts +1 -0
- package/dist/test/ix-grid-column-filter.test.js +36 -0
- package/dist/test/ix-grid-column-filter.test.js.map +1 -0
- package/dist/test/ix-grid-row-filter.test.d.ts +1 -0
- package/dist/test/ix-grid-row-filter.test.js +37 -0
- package/dist/test/ix-grid-row-filter.test.js.map +1 -0
- package/dist/test/ix-grid.test.d.ts +1 -0
- package/dist/test/ix-grid.test.js +43 -0
- package/dist/test/ix-grid.test.js.map +1 -0
- package/dist/test/ix-pagination.test.d.ts +1 -0
- package/dist/test/ix-pagination.test.js +28 -0
- package/dist/test/ix-pagination.test.js.map +1 -0
- package/package.json +95 -0
- package/src/IxGrid.ts +229 -0
- package/src/components/IxGridColumnFilter.ts +144 -0
- package/src/components/IxGridRowFilter.ts +352 -0
- package/src/components/IxPagination.ts +85 -0
- package/src/components/grid-column-filter-styles.ts +71 -0
- package/src/components/grid-row-filter-styles.ts +414 -0
- package/src/components/pagination-styles.ts +33 -0
- package/src/grid-view-styles.ts +265 -0
- package/src/index.ts +1 -0
- package/src/ix-grid-copy.ts +10 -0
- package/src/ix-grid.ts +3 -0
- package/src/test/ix-grid-column-filter.test.ts +47 -0
- package/src/test/ix-grid-row-filter.test.ts +48 -0
- package/src/test/ix-grid.test.ts +50 -0
- package/src/test/ix-pagination.test.ts +33 -0
- package/tsconfig.json +22 -0
- package/web-dev-server.config.mjs +27 -0
- package/web-test-runner.config.mjs +43 -0
package/dist/IxGrid.js
ADDED
|
@@ -0,0 +1,227 @@
|
|
|
1
|
+
import { __decorate } from "tslib";
|
|
2
|
+
import { html, LitElement, nothing } from 'lit';
|
|
3
|
+
import { property, state } from 'lit/decorators.js';
|
|
4
|
+
import { ifDefined } from 'lit/directives/if-defined.js';
|
|
5
|
+
import '@vaadin/grid';
|
|
6
|
+
import { columnBodyRenderer, columnHeaderRenderer } from '@vaadin/grid/lit.js';
|
|
7
|
+
import '@digital-realty/ix-icon-button/ix-icon-button.js';
|
|
8
|
+
import '@digital-realty/ix-icon/ix-icon.js';
|
|
9
|
+
import './components/IxPagination.js';
|
|
10
|
+
import './components/IxGridColumnFilter.js';
|
|
11
|
+
import './components/IxGridRowFilter.js';
|
|
12
|
+
import { IxGridViewStyles } from './grid-view-styles.js';
|
|
13
|
+
import { copy } from './ix-grid-copy.js';
|
|
14
|
+
export class IxGrid extends LitElement {
|
|
15
|
+
constructor() {
|
|
16
|
+
super(...arguments);
|
|
17
|
+
this.columns = [];
|
|
18
|
+
this.rows = [];
|
|
19
|
+
this.defaultEmptyText = 'No data to display';
|
|
20
|
+
this.sortedColumn = '';
|
|
21
|
+
this.sortDirection = '';
|
|
22
|
+
this.hideHeader = false;
|
|
23
|
+
this.rowLimit = 0;
|
|
24
|
+
this.page = 1;
|
|
25
|
+
this.pageSize = 10;
|
|
26
|
+
this.pageSizes = [5, 10, 25, 100];
|
|
27
|
+
this.recordCount = 0;
|
|
28
|
+
this.localStorageID = undefined;
|
|
29
|
+
this.filters = [];
|
|
30
|
+
this.isLoading = false;
|
|
31
|
+
this.isExpanded = false;
|
|
32
|
+
this.renderColumnHeader = (column) => html `
|
|
33
|
+
<div
|
|
34
|
+
@click=${() => column.sortable && this.handleSort(column.name)}
|
|
35
|
+
@keyDown=${() => column.sortable && this.handleSort(column.name)}
|
|
36
|
+
class="header"
|
|
37
|
+
>
|
|
38
|
+
${column.header}
|
|
39
|
+
${column.sortable
|
|
40
|
+
? html `<ix-icon title="Sort" class="header-sort-icon"
|
|
41
|
+
>${this.sortDirection === 'desc' &&
|
|
42
|
+
this.sortedColumn === column.name
|
|
43
|
+
? `arrow_upward`
|
|
44
|
+
: `arrow_downward`}</ix-icon
|
|
45
|
+
>`
|
|
46
|
+
: nothing}
|
|
47
|
+
</div>
|
|
48
|
+
`;
|
|
49
|
+
this.renderHeader = () => html `
|
|
50
|
+
<div class="grid-header">
|
|
51
|
+
<h2>
|
|
52
|
+
${html `${this.recordCount}
|
|
53
|
+
${copy.user}${this.recordCount === 1 ? '' : 's'}`}
|
|
54
|
+
</h2>
|
|
55
|
+
<div class="grid-menu">
|
|
56
|
+
<ix-grid-column-filter
|
|
57
|
+
.columns=${this.columns}
|
|
58
|
+
localStorageID=${ifDefined(this.localStorageID)}
|
|
59
|
+
@columnFilter=${(e) => {
|
|
60
|
+
e.detail.columns.forEach((column, id) => {
|
|
61
|
+
this.columns[id].hidden = column.hidden;
|
|
62
|
+
});
|
|
63
|
+
this.updatePage();
|
|
64
|
+
}}
|
|
65
|
+
></ix-grid-column-filter>
|
|
66
|
+
<ix-icon-button icon="download"></ix-icon-button>
|
|
67
|
+
<ix-grid-row-filter
|
|
68
|
+
.columns=${this.columns}
|
|
69
|
+
@rowFilter=${(e) => {
|
|
70
|
+
this.filters = e.detail.filters;
|
|
71
|
+
this.updatePage();
|
|
72
|
+
}}
|
|
73
|
+
></ix-grid-row-filter>
|
|
74
|
+
</div>
|
|
75
|
+
</div>
|
|
76
|
+
`;
|
|
77
|
+
this.renderRowLimitControls = () => {
|
|
78
|
+
if (this.rows.length <= this.rowLimit)
|
|
79
|
+
return nothing;
|
|
80
|
+
return html `
|
|
81
|
+
<div class="row-limit">
|
|
82
|
+
<ix-button
|
|
83
|
+
appearance="text"
|
|
84
|
+
@click=${() => {
|
|
85
|
+
this.isExpanded = !this.isExpanded;
|
|
86
|
+
}}
|
|
87
|
+
has-icon
|
|
88
|
+
>
|
|
89
|
+
${this.isExpanded ? copy.viewLess : copy.viewMore}
|
|
90
|
+
<ix-icon slot="icon">${this.isExpanded ? 'remove' : 'add'}</ix-icon>
|
|
91
|
+
</ix-button>
|
|
92
|
+
</div>
|
|
93
|
+
`;
|
|
94
|
+
};
|
|
95
|
+
this.renderPaginationControls = () => html `
|
|
96
|
+
<ix-pagination
|
|
97
|
+
.page=${this.page}
|
|
98
|
+
.pageSize=${this.pageSize}
|
|
99
|
+
.pageSizes=${this.pageSizes}
|
|
100
|
+
.recordCount=${this.recordCount}
|
|
101
|
+
@updatePagination=${(e) => {
|
|
102
|
+
this.page = e.detail.page;
|
|
103
|
+
this.pageSize = e.detail.pageSize;
|
|
104
|
+
this.updatePage();
|
|
105
|
+
}}
|
|
106
|
+
></ix-pagination>
|
|
107
|
+
`;
|
|
108
|
+
}
|
|
109
|
+
get columnNames() {
|
|
110
|
+
return this.columns.map((column) => column.name);
|
|
111
|
+
}
|
|
112
|
+
async updatePage() {
|
|
113
|
+
const filters = this.filters.reduce((columnFilters, { columnField, value }) => ({
|
|
114
|
+
...columnFilters,
|
|
115
|
+
[columnField]: value,
|
|
116
|
+
}), {});
|
|
117
|
+
const urlParams = {
|
|
118
|
+
sort: this.sortedColumn,
|
|
119
|
+
order: this.sortDirection,
|
|
120
|
+
page: this.page.toString(),
|
|
121
|
+
size: this.pageSize.toString(),
|
|
122
|
+
...filters,
|
|
123
|
+
};
|
|
124
|
+
const url = new URL(window.location.href);
|
|
125
|
+
const searchParams = new URLSearchParams(urlParams);
|
|
126
|
+
url.search = searchParams.toString();
|
|
127
|
+
window.history.replaceState(null, '', url.toString());
|
|
128
|
+
this.dispatchEvent(new CustomEvent('change', {
|
|
129
|
+
detail: {
|
|
130
|
+
columnName: this.sortedColumn,
|
|
131
|
+
sortOrder: this.sortDirection,
|
|
132
|
+
page: this.page,
|
|
133
|
+
pageSize: this.pageSize,
|
|
134
|
+
filters,
|
|
135
|
+
},
|
|
136
|
+
bubbles: true,
|
|
137
|
+
composed: true,
|
|
138
|
+
}));
|
|
139
|
+
}
|
|
140
|
+
handleSort(column = '') {
|
|
141
|
+
if (this.sortedColumn !== column) {
|
|
142
|
+
this.sortDirection = 'asc';
|
|
143
|
+
}
|
|
144
|
+
else {
|
|
145
|
+
this.sortDirection = this.sortDirection === 'asc' ? 'desc' : 'asc';
|
|
146
|
+
}
|
|
147
|
+
this.sortedColumn = column;
|
|
148
|
+
this.updatePage();
|
|
149
|
+
}
|
|
150
|
+
render() {
|
|
151
|
+
var _a;
|
|
152
|
+
return html `
|
|
153
|
+
<div class=${`grid-container ${this.isLoading ? 'loading' : ''}`}>
|
|
154
|
+
${this.hideHeader ? nothing : this.renderHeader()}
|
|
155
|
+
<vaadin-grid
|
|
156
|
+
.items=${this.rowLimit > 0 && !this.isExpanded
|
|
157
|
+
? this.rows.slice(0, this.rowLimit)
|
|
158
|
+
: this.rows}
|
|
159
|
+
all-rows-visible
|
|
160
|
+
column-reordering-allowed
|
|
161
|
+
theme="no-border"
|
|
162
|
+
>
|
|
163
|
+
${(_a = this.columns) === null || _a === void 0 ? void 0 : _a.map((column) => column.name
|
|
164
|
+
? html `<vaadin-grid-column
|
|
165
|
+
${columnHeaderRenderer(() => this.renderColumnHeader(column), this.sortDirection)}
|
|
166
|
+
${columnBodyRenderer(column.bodyRenderer, [])}
|
|
167
|
+
resizable
|
|
168
|
+
width=${ifDefined(column.width)}
|
|
169
|
+
?hidden=${column.hidden}
|
|
170
|
+
?frozen-to-end=${column.frozenToEnd}
|
|
171
|
+
></vaadin-grid-column>`
|
|
172
|
+
: nothing)}
|
|
173
|
+
</vaadin-grid>
|
|
174
|
+
${this.rowLimit > 0
|
|
175
|
+
? this.renderRowLimitControls()
|
|
176
|
+
: this.renderPaginationControls()}
|
|
177
|
+
</div>
|
|
178
|
+
`;
|
|
179
|
+
}
|
|
180
|
+
}
|
|
181
|
+
IxGrid.styles = [IxGridViewStyles];
|
|
182
|
+
__decorate([
|
|
183
|
+
property({ type: Array })
|
|
184
|
+
], IxGrid.prototype, "columns", void 0);
|
|
185
|
+
__decorate([
|
|
186
|
+
property({ type: Array })
|
|
187
|
+
], IxGrid.prototype, "rows", void 0);
|
|
188
|
+
__decorate([
|
|
189
|
+
property({ type: String })
|
|
190
|
+
], IxGrid.prototype, "defaultEmptyText", void 0);
|
|
191
|
+
__decorate([
|
|
192
|
+
property({ type: String })
|
|
193
|
+
], IxGrid.prototype, "sortedColumn", void 0);
|
|
194
|
+
__decorate([
|
|
195
|
+
property({ type: String })
|
|
196
|
+
], IxGrid.prototype, "sortDirection", void 0);
|
|
197
|
+
__decorate([
|
|
198
|
+
property({ type: Boolean })
|
|
199
|
+
], IxGrid.prototype, "hideHeader", void 0);
|
|
200
|
+
__decorate([
|
|
201
|
+
property({ type: Number })
|
|
202
|
+
], IxGrid.prototype, "rowLimit", void 0);
|
|
203
|
+
__decorate([
|
|
204
|
+
property({ type: Number })
|
|
205
|
+
], IxGrid.prototype, "page", void 0);
|
|
206
|
+
__decorate([
|
|
207
|
+
property({ type: Number })
|
|
208
|
+
], IxGrid.prototype, "pageSize", void 0);
|
|
209
|
+
__decorate([
|
|
210
|
+
property({ type: Array })
|
|
211
|
+
], IxGrid.prototype, "pageSizes", void 0);
|
|
212
|
+
__decorate([
|
|
213
|
+
property({ type: Number })
|
|
214
|
+
], IxGrid.prototype, "recordCount", void 0);
|
|
215
|
+
__decorate([
|
|
216
|
+
property({ type: String })
|
|
217
|
+
], IxGrid.prototype, "localStorageID", void 0);
|
|
218
|
+
__decorate([
|
|
219
|
+
state()
|
|
220
|
+
], IxGrid.prototype, "filters", void 0);
|
|
221
|
+
__decorate([
|
|
222
|
+
state()
|
|
223
|
+
], IxGrid.prototype, "isLoading", void 0);
|
|
224
|
+
__decorate([
|
|
225
|
+
state()
|
|
226
|
+
], IxGrid.prototype, "isExpanded", void 0);
|
|
227
|
+
//# sourceMappingURL=IxGrid.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"IxGrid.js","sourceRoot":"","sources":["../src/IxGrid.ts"],"names":[],"mappings":"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","sourcesContent":["import { html, LitElement, nothing } from 'lit';\nimport { property, state } from 'lit/decorators.js';\nimport { ifDefined } from 'lit/directives/if-defined.js';\nimport '@vaadin/grid';\nimport { columnBodyRenderer, columnHeaderRenderer } from '@vaadin/grid/lit.js';\nimport '@digital-realty/ix-icon-button/ix-icon-button.js';\nimport '@digital-realty/ix-icon/ix-icon.js';\nimport './components/IxPagination.js';\nimport './components/IxGridColumnFilter.js';\nimport './components/IxGridRowFilter.js';\nimport type { Filter } from './components/IxGridRowFilter.js';\nimport { IxGridViewStyles } from './grid-view-styles.js';\nimport { copy } from './ix-grid-copy.js';\n\nexport interface Row {\n [key: string]: string;\n}\n\nexport interface Column {\n name: string;\n header: string;\n bodyRenderer: (row: Row) => any;\n width?: string;\n sortable?: boolean;\n filterable?: boolean;\n hidden?: boolean;\n frozenToEnd?: boolean;\n}\n\nexport class IxGrid extends LitElement {\n static readonly styles = [IxGridViewStyles];\n\n @property({ type: Array }) columns: Column[] = [];\n\n @property({ type: Array }) rows: Row[] = [];\n\n @property({ type: String }) defaultEmptyText = 'No data to display';\n\n @property({ type: String }) sortedColumn = '';\n\n @property({ type: String }) sortDirection = '';\n\n @property({ type: Boolean }) hideHeader = false;\n\n @property({ type: Number }) rowLimit: number = 0;\n\n @property({ type: Number }) page = 1;\n\n @property({ type: Number }) pageSize = 10;\n\n @property({ type: Array }) pageSizes: number[] = [5, 10, 25, 100];\n\n @property({ type: Number }) recordCount = 0;\n\n @property({ type: String }) localStorageID: string | undefined = undefined;\n\n @state() private filters: Filter[] = [];\n\n @state() isLoading = false;\n\n @state() isExpanded = false;\n\n get columnNames() {\n return this.columns.map((column: Column) => column.name);\n }\n\n private async updatePage() {\n const filters = this.filters.reduce(\n (columnFilters: { [key: string]: string }, { columnField, value }) => ({\n ...columnFilters,\n [columnField]: value,\n }),\n {}\n );\n\n const urlParams: { [key: string]: string } = {\n sort: this.sortedColumn,\n order: this.sortDirection,\n page: this.page.toString(),\n size: this.pageSize.toString(),\n ...filters,\n };\n\n const url = new URL(window.location.href);\n const searchParams = new URLSearchParams(urlParams);\n url.search = searchParams.toString();\n window.history.replaceState(null, '', url.toString());\n\n this.dispatchEvent(\n new CustomEvent('change', {\n detail: {\n columnName: this.sortedColumn,\n sortOrder: this.sortDirection,\n page: this.page,\n pageSize: this.pageSize,\n filters,\n },\n bubbles: true,\n composed: true,\n })\n );\n }\n\n handleSort(column: string = '') {\n if (this.sortedColumn !== column) {\n this.sortDirection = 'asc';\n } else {\n this.sortDirection = this.sortDirection === 'asc' ? 'desc' : 'asc';\n }\n this.sortedColumn = column;\n\n this.updatePage();\n }\n\n private renderColumnHeader = (column: Column) => html`\n <div\n @click=${() => column.sortable && this.handleSort(column.name)}\n @keyDown=${() => column.sortable && this.handleSort(column.name)}\n class=\"header\"\n >\n ${column.header}\n ${column.sortable\n ? html`<ix-icon title=\"Sort\" class=\"header-sort-icon\"\n >${this.sortDirection === 'desc' &&\n this.sortedColumn === column.name\n ? `arrow_upward`\n : `arrow_downward`}</ix-icon\n >`\n : nothing}\n </div>\n `;\n\n private renderHeader = () => html`\n <div class=\"grid-header\">\n <h2>\n ${html`${this.recordCount}\n ${copy.user}${this.recordCount === 1 ? '' : 's'}`}\n </h2>\n <div class=\"grid-menu\">\n <ix-grid-column-filter\n .columns=${this.columns}\n localStorageID=${ifDefined(this.localStorageID)}\n @columnFilter=${(e: CustomEvent) => {\n e.detail.columns.forEach((column: Column, id: number) => {\n this.columns[id].hidden = column.hidden;\n });\n this.updatePage();\n }}\n ></ix-grid-column-filter>\n <ix-icon-button icon=\"download\"></ix-icon-button>\n <ix-grid-row-filter\n .columns=${this.columns}\n @rowFilter=${(e: CustomEvent) => {\n this.filters = e.detail.filters;\n this.updatePage();\n }}\n ></ix-grid-row-filter>\n </div>\n </div>\n `;\n\n private renderRowLimitControls = () => {\n if (this.rows.length <= this.rowLimit) return nothing;\n\n return html`\n <div class=\"row-limit\">\n <ix-button\n appearance=\"text\"\n @click=${() => {\n this.isExpanded = !this.isExpanded;\n }}\n has-icon\n >\n ${this.isExpanded ? copy.viewLess : copy.viewMore}\n <ix-icon slot=\"icon\">${this.isExpanded ? 'remove' : 'add'}</ix-icon>\n </ix-button>\n </div>\n `;\n };\n\n private renderPaginationControls = () => html`\n <ix-pagination\n .page=${this.page}\n .pageSize=${this.pageSize}\n .pageSizes=${this.pageSizes}\n .recordCount=${this.recordCount}\n @updatePagination=${(e: CustomEvent) => {\n this.page = e.detail.page;\n this.pageSize = e.detail.pageSize;\n this.updatePage();\n }}\n ></ix-pagination>\n `;\n\n render() {\n return html`\n <div class=${`grid-container ${this.isLoading ? 'loading' : ''}`}>\n ${this.hideHeader ? nothing : this.renderHeader()}\n <vaadin-grid\n .items=${this.rowLimit > 0 && !this.isExpanded\n ? this.rows.slice(0, this.rowLimit)\n : this.rows}\n all-rows-visible\n column-reordering-allowed\n theme=\"no-border\"\n >\n ${this.columns?.map((column: Column) =>\n column.name\n ? html`<vaadin-grid-column\n ${columnHeaderRenderer(\n () => this.renderColumnHeader(column),\n this.sortDirection\n )}\n ${columnBodyRenderer(column.bodyRenderer, [])}\n resizable\n width=${ifDefined(column.width)}\n ?hidden=${column.hidden}\n ?frozen-to-end=${column.frozenToEnd}\n ></vaadin-grid-column>`\n : nothing\n )}\n </vaadin-grid>\n ${this.rowLimit > 0\n ? this.renderRowLimitControls()\n : this.renderPaginationControls()}\n </div>\n `;\n }\n}\n"]}
|
|
@@ -0,0 +1,20 @@
|
|
|
1
|
+
import { LitElement } from 'lit';
|
|
2
|
+
import '@digital-realty/ix-icon-button/ix-icon-button.js';
|
|
3
|
+
import '@digital-realty/ix-select/ix-select.js';
|
|
4
|
+
import '@digital-realty/ix-switch/ix-switch.js';
|
|
5
|
+
import type { Column } from '../IxGrid.js';
|
|
6
|
+
export declare class IxGridColumnFilter extends LitElement {
|
|
7
|
+
static readonly styles: import("lit").CSSResult[];
|
|
8
|
+
columns: Column[];
|
|
9
|
+
localStorageID: string | undefined;
|
|
10
|
+
private isDropdownVisible;
|
|
11
|
+
disabledColumns: string[];
|
|
12
|
+
connectedCallback(): void;
|
|
13
|
+
disconnectedCallback(): void;
|
|
14
|
+
outerInteraction: (e: Event) => void;
|
|
15
|
+
initializeLocalStorage(): void;
|
|
16
|
+
toggleColumn(id: number): void;
|
|
17
|
+
updateColumn(e: Event, id: number): void;
|
|
18
|
+
dispatchUpdate(columns?: Column[]): void;
|
|
19
|
+
render(): import("lit").TemplateResult<1>;
|
|
20
|
+
}
|
|
@@ -0,0 +1,133 @@
|
|
|
1
|
+
import { __decorate } from "tslib";
|
|
2
|
+
import { LitElement, html, nothing } from 'lit';
|
|
3
|
+
import { customElement, property, state } from 'lit/decorators.js';
|
|
4
|
+
import '@digital-realty/ix-icon-button/ix-icon-button.js';
|
|
5
|
+
import '@digital-realty/ix-select/ix-select.js';
|
|
6
|
+
import '@digital-realty/ix-switch/ix-switch.js';
|
|
7
|
+
import { IxGridViewStyles } from '../grid-view-styles.js';
|
|
8
|
+
import { IxGridColumnFilterStyles } from './grid-column-filter-styles.js';
|
|
9
|
+
const triggerKeys = [' ', 'Enter'];
|
|
10
|
+
let IxGridColumnFilter = class IxGridColumnFilter extends LitElement {
|
|
11
|
+
constructor() {
|
|
12
|
+
super(...arguments);
|
|
13
|
+
this.columns = [];
|
|
14
|
+
this.localStorageID = undefined;
|
|
15
|
+
this.isDropdownVisible = false;
|
|
16
|
+
this.disabledColumns = [];
|
|
17
|
+
this.outerInteraction = (e) => {
|
|
18
|
+
if (!e.composedPath().includes(this)) {
|
|
19
|
+
this.isDropdownVisible = false;
|
|
20
|
+
}
|
|
21
|
+
};
|
|
22
|
+
}
|
|
23
|
+
connectedCallback() {
|
|
24
|
+
super.connectedCallback();
|
|
25
|
+
document.addEventListener('click', this.outerInteraction);
|
|
26
|
+
this.initializeLocalStorage();
|
|
27
|
+
}
|
|
28
|
+
disconnectedCallback() {
|
|
29
|
+
super.disconnectedCallback();
|
|
30
|
+
document.removeEventListener('click', this.outerInteraction);
|
|
31
|
+
}
|
|
32
|
+
initializeLocalStorage() {
|
|
33
|
+
if (this.localStorageID) {
|
|
34
|
+
const storedDisabledColumns = localStorage.getItem(this.localStorageID);
|
|
35
|
+
if (storedDisabledColumns) {
|
|
36
|
+
const disabledColumns = JSON.parse(storedDisabledColumns);
|
|
37
|
+
this.columns.forEach((column, id) => {
|
|
38
|
+
if (disabledColumns.includes(column.name)) {
|
|
39
|
+
this.columns[id].hidden = true;
|
|
40
|
+
}
|
|
41
|
+
});
|
|
42
|
+
}
|
|
43
|
+
this.dispatchUpdate();
|
|
44
|
+
}
|
|
45
|
+
}
|
|
46
|
+
toggleColumn(id) {
|
|
47
|
+
this.columns[id].hidden = !this.columns[id].hidden;
|
|
48
|
+
this.disabledColumns = this.columns
|
|
49
|
+
.filter((column) => column.hidden)
|
|
50
|
+
.map((column) => column.name);
|
|
51
|
+
if (this.localStorageID !== undefined) {
|
|
52
|
+
localStorage.setItem(this.localStorageID, JSON.stringify(this.disabledColumns));
|
|
53
|
+
}
|
|
54
|
+
this.dispatchUpdate();
|
|
55
|
+
}
|
|
56
|
+
updateColumn(e, id) {
|
|
57
|
+
var _a;
|
|
58
|
+
const input = e.target;
|
|
59
|
+
const el = (_a = input.shadowRoot) === null || _a === void 0 ? void 0 : _a.querySelector('input');
|
|
60
|
+
this.columns[id].hidden = !(el === null || el === void 0 ? void 0 : el.checked);
|
|
61
|
+
this.disabledColumns = this.columns
|
|
62
|
+
.filter((column) => column.hidden)
|
|
63
|
+
.map((column) => column.name);
|
|
64
|
+
if (this.localStorageID !== undefined) {
|
|
65
|
+
localStorage.setItem(this.localStorageID, JSON.stringify(this.disabledColumns));
|
|
66
|
+
}
|
|
67
|
+
this.dispatchUpdate();
|
|
68
|
+
}
|
|
69
|
+
dispatchUpdate(columns = this.columns) {
|
|
70
|
+
this.dispatchEvent(new CustomEvent('columnFilter', {
|
|
71
|
+
detail: {
|
|
72
|
+
columns,
|
|
73
|
+
},
|
|
74
|
+
bubbles: true,
|
|
75
|
+
composed: true,
|
|
76
|
+
}));
|
|
77
|
+
}
|
|
78
|
+
render() {
|
|
79
|
+
var _a;
|
|
80
|
+
return html ` <div class="grid-menu">
|
|
81
|
+
<span
|
|
82
|
+
@click=${() => {
|
|
83
|
+
this.isDropdownVisible = true;
|
|
84
|
+
}}
|
|
85
|
+
@keyDown=${(e) => {
|
|
86
|
+
if (triggerKeys.includes(e.key)) {
|
|
87
|
+
this.isDropdownVisible = true;
|
|
88
|
+
}
|
|
89
|
+
}}
|
|
90
|
+
class="list list-dropdown"
|
|
91
|
+
>
|
|
92
|
+
<ix-icon-button icon="list"></ix-icon-button>
|
|
93
|
+
${this.disabledColumns.length > 0
|
|
94
|
+
? html `<div class="active"></div>`
|
|
95
|
+
: nothing}
|
|
96
|
+
${this.isDropdownVisible
|
|
97
|
+
? html ` <div class="dropdown-content">
|
|
98
|
+
${(_a = this.columns) === null || _a === void 0 ? void 0 : _a.map((col, id) => html `<div
|
|
99
|
+
class=${col.hidden ? 'disabled' : ''}
|
|
100
|
+
>
|
|
101
|
+
<label class="ix-switch-label">
|
|
102
|
+
<ix-switch
|
|
103
|
+
.selected=${!col.hidden}
|
|
104
|
+
@change=${(e) => this.updateColumn(e, id)}
|
|
105
|
+
>
|
|
106
|
+
</ix-switch>
|
|
107
|
+
<p>${col.header}</p>
|
|
108
|
+
</label>
|
|
109
|
+
</div>`)}
|
|
110
|
+
</div>`
|
|
111
|
+
: nothing}
|
|
112
|
+
</span>
|
|
113
|
+
</div>`;
|
|
114
|
+
}
|
|
115
|
+
};
|
|
116
|
+
IxGridColumnFilter.styles = [IxGridViewStyles, IxGridColumnFilterStyles];
|
|
117
|
+
__decorate([
|
|
118
|
+
property({ type: Array })
|
|
119
|
+
], IxGridColumnFilter.prototype, "columns", void 0);
|
|
120
|
+
__decorate([
|
|
121
|
+
property({ type: String })
|
|
122
|
+
], IxGridColumnFilter.prototype, "localStorageID", void 0);
|
|
123
|
+
__decorate([
|
|
124
|
+
state()
|
|
125
|
+
], IxGridColumnFilter.prototype, "isDropdownVisible", void 0);
|
|
126
|
+
__decorate([
|
|
127
|
+
state()
|
|
128
|
+
], IxGridColumnFilter.prototype, "disabledColumns", void 0);
|
|
129
|
+
IxGridColumnFilter = __decorate([
|
|
130
|
+
customElement('ix-grid-column-filter')
|
|
131
|
+
], IxGridColumnFilter);
|
|
132
|
+
export { IxGridColumnFilter };
|
|
133
|
+
//# sourceMappingURL=IxGridColumnFilter.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"IxGridColumnFilter.js","sourceRoot":"","sources":["../../src/components/IxGridColumnFilter.ts"],"names":[],"mappings":"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","sourcesContent":["import { LitElement, html, nothing } from 'lit';\nimport { customElement, property, state } from 'lit/decorators.js';\nimport '@digital-realty/ix-icon-button/ix-icon-button.js';\nimport '@digital-realty/ix-select/ix-select.js';\nimport '@digital-realty/ix-switch/ix-switch.js';\nimport { IxGridViewStyles } from '../grid-view-styles.js';\nimport { IxGridColumnFilterStyles } from './grid-column-filter-styles.js';\nimport type { Column } from '../IxGrid.js';\n\nconst triggerKeys = [' ', 'Enter'];\n\n@customElement('ix-grid-column-filter')\nexport class IxGridColumnFilter extends LitElement {\n static readonly styles = [IxGridViewStyles, IxGridColumnFilterStyles];\n\n @property({ type: Array }) columns: Column[] = [];\n\n @property({ type: String }) localStorageID: string | undefined = undefined;\n\n @state() private isDropdownVisible: boolean = false;\n\n @state() disabledColumns: string[] = [];\n\n connectedCallback() {\n super.connectedCallback();\n document.addEventListener('click', this.outerInteraction);\n this.initializeLocalStorage();\n }\n\n disconnectedCallback() {\n super.disconnectedCallback();\n document.removeEventListener('click', this.outerInteraction);\n }\n\n outerInteraction = (e: Event) => {\n if (!e.composedPath().includes(this)) {\n this.isDropdownVisible = false;\n }\n };\n\n initializeLocalStorage() {\n if (this.localStorageID) {\n const storedDisabledColumns = localStorage.getItem(this.localStorageID);\n\n if (storedDisabledColumns) {\n const disabledColumns = JSON.parse(storedDisabledColumns);\n this.columns.forEach((column, id) => {\n if (disabledColumns.includes(column.name)) {\n this.columns[id].hidden = true;\n }\n });\n }\n this.dispatchUpdate();\n }\n }\n\n toggleColumn(id: number) {\n this.columns[id].hidden = !this.columns[id].hidden;\n\n this.disabledColumns = this.columns\n .filter((column: Column) => column.hidden)\n .map((column: Column) => column.name);\n\n if (this.localStorageID !== undefined) {\n localStorage.setItem(\n this.localStorageID,\n JSON.stringify(this.disabledColumns)\n );\n }\n\n this.dispatchUpdate();\n }\n\n updateColumn(e: Event, id: number) {\n const input = e.target as HTMLElement;\n const el = input.shadowRoot?.querySelector('input');\n\n this.columns[id].hidden = !el?.checked;\n\n this.disabledColumns = this.columns\n .filter((column: Column) => column.hidden)\n .map((column: Column) => column.name);\n\n if (this.localStorageID !== undefined) {\n localStorage.setItem(\n this.localStorageID,\n JSON.stringify(this.disabledColumns)\n );\n }\n\n this.dispatchUpdate();\n }\n\n dispatchUpdate(columns = this.columns) {\n this.dispatchEvent(\n new CustomEvent('columnFilter', {\n detail: {\n columns,\n },\n bubbles: true,\n composed: true,\n })\n );\n }\n\n render() {\n return html` <div class=\"grid-menu\">\n <span\n @click=${() => {\n this.isDropdownVisible = true;\n }}\n @keyDown=${(e: KeyboardEvent) => {\n if (triggerKeys.includes(e.key)) {\n this.isDropdownVisible = true;\n }\n }}\n class=\"list list-dropdown\"\n >\n <ix-icon-button icon=\"list\"></ix-icon-button>\n ${this.disabledColumns.length > 0\n ? html`<div class=\"active\"></div>`\n : nothing}\n ${this.isDropdownVisible\n ? html` <div class=\"dropdown-content\">\n ${this.columns?.map(\n (col: Column, id: number) => html`<div\n class=${col.hidden ? 'disabled' : ''}\n >\n <label class=\"ix-switch-label\">\n <ix-switch\n .selected=${!col.hidden}\n @change=${(e: Event) => this.updateColumn(e, id)}\n >\n </ix-switch>\n <p>${col.header}</p>\n </label>\n </div>`\n )}\n </div>`\n : nothing}\n </span>\n </div>`;\n }\n}\n"]}
|
|
@@ -0,0 +1,37 @@
|
|
|
1
|
+
import { LitElement } from 'lit';
|
|
2
|
+
import '@digital-realty/ix-icon-button/ix-icon-button.js';
|
|
3
|
+
import '@digital-realty/ix-select/ix-select.js';
|
|
4
|
+
import type { Column } from '../IxGrid.js';
|
|
5
|
+
export interface Filter {
|
|
6
|
+
columnField: string;
|
|
7
|
+
operatorValue: string;
|
|
8
|
+
value: string;
|
|
9
|
+
}
|
|
10
|
+
export declare class IxGridRowFilter extends LitElement {
|
|
11
|
+
static readonly styles: import("lit").CSSResult[];
|
|
12
|
+
columns: Column[];
|
|
13
|
+
private isDropdownVisible;
|
|
14
|
+
private filters;
|
|
15
|
+
private filterableColumns;
|
|
16
|
+
private filterColumns;
|
|
17
|
+
private activeFilters;
|
|
18
|
+
private mapSelect;
|
|
19
|
+
updateActiveFilters(): void;
|
|
20
|
+
connectedCallback(): void;
|
|
21
|
+
disconnectedCallback(): void;
|
|
22
|
+
firstUpdated(): void;
|
|
23
|
+
get filterNames(): string[];
|
|
24
|
+
get unselectedFilters(): string[];
|
|
25
|
+
closeOnOuterClick: (e: Event) => void;
|
|
26
|
+
parseFilterQueryString(): Filter[];
|
|
27
|
+
dispatchUpdate(): void;
|
|
28
|
+
addFilter(): void;
|
|
29
|
+
clearFilters(): void;
|
|
30
|
+
removeFilter(index: number): void;
|
|
31
|
+
private onfilterColumnChange;
|
|
32
|
+
private onfilterValueChange;
|
|
33
|
+
renderToolTip(): string | import("lit").TemplateResult<1>;
|
|
34
|
+
renderFilterInput(value: Filter, index: number): import("lit").TemplateResult<1>;
|
|
35
|
+
renderDropdown(): import("lit").TemplateResult<1>;
|
|
36
|
+
render(): import("lit").TemplateResult<1>;
|
|
37
|
+
}
|