@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.
Files changed (69) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +62 -0
  3. package/demo/columns.js +65 -0
  4. package/demo/contacts.js +292 -0
  5. package/demo/index.html +92 -0
  6. package/dist/IxGrid.d.ts +46 -0
  7. package/dist/IxGrid.js +227 -0
  8. package/dist/IxGrid.js.map +1 -0
  9. package/dist/components/IxGridColumnFilter.d.ts +20 -0
  10. package/dist/components/IxGridColumnFilter.js +133 -0
  11. package/dist/components/IxGridColumnFilter.js.map +1 -0
  12. package/dist/components/IxGridRowFilter.d.ts +37 -0
  13. package/dist/components/IxGridRowFilter.js +317 -0
  14. package/dist/components/IxGridRowFilter.js.map +1 -0
  15. package/dist/components/IxPagination.d.ts +13 -0
  16. package/dist/components/IxPagination.js +93 -0
  17. package/dist/components/IxPagination.js.map +1 -0
  18. package/dist/components/grid-column-filter-styles.d.ts +1 -0
  19. package/dist/components/grid-column-filter-styles.js +71 -0
  20. package/dist/components/grid-column-filter-styles.js.map +1 -0
  21. package/dist/components/grid-row-filter-styles.d.ts +1 -0
  22. package/dist/components/grid-row-filter-styles.js +414 -0
  23. package/dist/components/grid-row-filter-styles.js.map +1 -0
  24. package/dist/components/pagination-styles.d.ts +1 -0
  25. package/dist/components/pagination-styles.js +33 -0
  26. package/dist/components/pagination-styles.js.map +1 -0
  27. package/dist/grid-view-styles.d.ts +1 -0
  28. package/dist/grid-view-styles.js +265 -0
  29. package/dist/grid-view-styles.js.map +1 -0
  30. package/dist/index.d.ts +1 -0
  31. package/dist/index.js +2 -0
  32. package/dist/index.js.map +1 -0
  33. package/dist/ix-grid-copy.d.ts +10 -0
  34. package/dist/ix-grid-copy.js +11 -0
  35. package/dist/ix-grid-copy.js.map +1 -0
  36. package/dist/ix-grid.d.ts +1 -0
  37. package/dist/ix-grid.js +3 -0
  38. package/dist/ix-grid.js.map +1 -0
  39. package/dist/test/ix-grid-column-filter.test.d.ts +1 -0
  40. package/dist/test/ix-grid-column-filter.test.js +36 -0
  41. package/dist/test/ix-grid-column-filter.test.js.map +1 -0
  42. package/dist/test/ix-grid-row-filter.test.d.ts +1 -0
  43. package/dist/test/ix-grid-row-filter.test.js +37 -0
  44. package/dist/test/ix-grid-row-filter.test.js.map +1 -0
  45. package/dist/test/ix-grid.test.d.ts +1 -0
  46. package/dist/test/ix-grid.test.js +43 -0
  47. package/dist/test/ix-grid.test.js.map +1 -0
  48. package/dist/test/ix-pagination.test.d.ts +1 -0
  49. package/dist/test/ix-pagination.test.js +28 -0
  50. package/dist/test/ix-pagination.test.js.map +1 -0
  51. package/package.json +95 -0
  52. package/src/IxGrid.ts +229 -0
  53. package/src/components/IxGridColumnFilter.ts +144 -0
  54. package/src/components/IxGridRowFilter.ts +352 -0
  55. package/src/components/IxPagination.ts +85 -0
  56. package/src/components/grid-column-filter-styles.ts +71 -0
  57. package/src/components/grid-row-filter-styles.ts +414 -0
  58. package/src/components/pagination-styles.ts +33 -0
  59. package/src/grid-view-styles.ts +265 -0
  60. package/src/index.ts +1 -0
  61. package/src/ix-grid-copy.ts +10 -0
  62. package/src/ix-grid.ts +3 -0
  63. package/src/test/ix-grid-column-filter.test.ts +47 -0
  64. package/src/test/ix-grid-row-filter.test.ts +48 -0
  65. package/src/test/ix-grid.test.ts +50 -0
  66. package/src/test/ix-pagination.test.ts +33 -0
  67. package/tsconfig.json +22 -0
  68. package/web-dev-server.config.mjs +27 -0
  69. 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
+ }