Skip to content

Commit 105226f

Browse files
committed
Fix build and update packages
1 parent ebe676b commit 105226f

10 files changed

+884
-2528
lines changed
+1-9
Original file line numberDiff line numberDiff line change
@@ -1,12 +1,10 @@
11
import { TestBed } from '@angular/core/testing';
22
import { AppComponent } from './app.component';
3-
import { NxWelcomeComponent } from './nx-welcome.component';
4-
import { RouterTestingModule } from '@angular/router/testing';
53

64
describe('AppComponent', () => {
75
beforeEach(async () => {
86
await TestBed.configureTestingModule({
9-
imports: [AppComponent, NxWelcomeComponent, RouterTestingModule],
7+
imports: [AppComponent],
108
}).compileComponents();
119
});
1210

@@ -16,10 +14,4 @@ describe('AppComponent', () => {
1614
const compiled = fixture.nativeElement as HTMLElement;
1715
expect(compiled.querySelector('h1')?.textContent).toContain('Welcome demo');
1816
});
19-
20-
it(`should have as title 'demo'`, () => {
21-
const fixture = TestBed.createComponent(AppComponent);
22-
const app = fixture.componentInstance;
23-
expect(app.title).toEqual('demo');
24-
});
2517
});

apps/demo/src/app/app.component.ts

+5-7
Original file line numberDiff line numberDiff line change
@@ -28,11 +28,9 @@ export class AppComponent {
2828
onRowClick = (event: any) => console.log('Grid row clicked', event);
2929

3030
columns = ['Name', 'Email', 'Phone Number'];
31-
data: TData = new Array(20)
32-
.fill(undefined)
33-
.map(() => [
34-
faker.person.fullName(),
35-
faker.internet.email(),
36-
faker.phone.number(),
37-
]);
31+
data: TData = Array.from({ length: 100 }).map(() => [
32+
faker.person.fullName(),
33+
faker.internet.email(),
34+
faker.phone.number(),
35+
]);
3836
}

package.json

+31-28
Original file line numberDiff line numberDiff line change
@@ -4,35 +4,38 @@
44
"license": "MIT",
55
"scripts": {
66
"build": "nx build gridjs-angular",
7-
"dev": "nx serve demo"
7+
"test": "nx test gridjs-angular",
8+
"dev": "nx serve demo",
9+
"publish": "cd dist/packages/gridjs-angular && pnpm publish --access public",
10+
"prepublishOnly": "pnpm run build"
811
},
912
"private": true,
1013
"dependencies": {
11-
"@angular/animations": "~17.1.2",
12-
"@angular/common": "~17.1.2",
13-
"@angular/compiler": "~17.1.2",
14-
"@angular/core": "~17.1.2",
15-
"@angular/forms": "~17.1.2",
16-
"@angular/platform-browser": "~17.1.2",
17-
"@angular/platform-browser-dynamic": "~17.1.2",
18-
"@angular/router": "~17.1.2",
14+
"@angular/animations": "~17.2.1",
15+
"@angular/common": "~17.2.1",
16+
"@angular/compiler": "~17.2.1",
17+
"@angular/core": "~17.2.1",
18+
"@angular/forms": "~17.2.1",
19+
"@angular/platform-browser": "~17.2.1",
20+
"@angular/platform-browser-dynamic": "~17.2.1",
21+
"@angular/router": "~17.2.1",
1922
"@nx/angular": "17.3.1",
2023
"gridjs": "^6.1.1",
2124
"rxjs": "~7.8.1",
2225
"tslib": "^2.6.2",
23-
"zone.js": "~0.14.3"
26+
"zone.js": "~0.14.4"
2427
},
2528
"devDependencies": {
26-
"@angular-devkit/build-angular": "~17.1.2",
27-
"@angular-devkit/core": "~17.1.2",
28-
"@angular-devkit/schematics": "~17.1.2",
29+
"@angular-devkit/build-angular": "~17.2.0",
30+
"@angular-devkit/core": "~17.2.0",
31+
"@angular-devkit/schematics": "~17.2.0",
2932
"@angular-eslint/eslint-plugin": "~17.2.1",
3033
"@angular-eslint/eslint-plugin-template": "~17.2.1",
3134
"@angular-eslint/template-parser": "~17.2.1",
32-
"@angular/cli": "~17.1.2",
33-
"@angular/compiler-cli": "~17.1.2",
34-
"@angular/language-service": "~17.1.2",
35-
"@faker-js/faker": "^8.4.0",
35+
"@angular/cli": "~17.2.0",
36+
"@angular/compiler-cli": "~17.2.1",
37+
"@angular/language-service": "~17.2.1",
38+
"@faker-js/faker": "^8.4.1",
3639
"@nx/devkit": "17.3.1",
3740
"@nx/eslint": "17.3.1",
3841
"@nx/eslint-plugin": "17.3.1",
@@ -41,29 +44,29 @@
4144
"@nx/playwright": "17.3.1",
4245
"@nx/workspace": "17.3.1",
4346
"@playwright/test": "^1.41.2",
44-
"@schematics/angular": "~17.1.2",
47+
"@schematics/angular": "~17.2.0",
4548
"@swc-node/register": "~1.8.0",
46-
"@swc/core": "~1.3.107",
47-
"@swc/helpers": "~0.5.3",
49+
"@swc/core": "~1.4.1",
50+
"@swc/helpers": "~0.5.6",
4851
"@types/jest": "^29.5.12",
4952
"@types/node": "18.16.9",
50-
"@typescript-eslint/eslint-plugin": "^6.20.0",
51-
"@typescript-eslint/parser": "^6.20.0",
53+
"@typescript-eslint/eslint-plugin": "^7.0.1",
54+
"@typescript-eslint/parser": "^7.0.1",
5255
"autoprefixer": "^10.4.17",
53-
"change-case": "^5.4.2",
56+
"change-case": "^5.4.3",
5457
"eslint": "~8.56.0",
5558
"eslint-config-prettier": "^9.1.0",
56-
"eslint-plugin-playwright": "^0.22.1",
59+
"eslint-plugin-playwright": "^1.1.2",
5760
"jest": "^29.7.0",
5861
"jest-environment-jsdom": "^29.7.0",
59-
"jest-preset-angular": "~14.0.0",
62+
"jest-preset-angular": "~14.0.3",
6063
"jsonc-eslint-parser": "^2.4.0",
6164
"mustache": "^4.2.0",
62-
"ng-packagr": "~17.1.2",
65+
"ng-packagr": "~17.2.0",
6366
"nx": "17.3.1",
64-
"postcss": "^8.4.33",
67+
"postcss": "^8.4.35",
6568
"postcss-url": "~10.1.3",
66-
"prettier": "^3.2.4",
69+
"prettier": "^3.2.5",
6770
"ts-jest": "^29.1.2",
6871
"ts-node": "10.9.2",
6972
"typescript": "~5.3.3"

packages/gridjs-angular/README.md

+3-14
Original file line numberDiff line numberDiff line change
@@ -13,26 +13,16 @@ npm install gridjs gridjs-angular
1313

1414
## Usage
1515

16-
In your module
17-
18-
```ts
19-
import { GridjsAngularModule } from 'gridjs-angular';
20-
21-
@NgModule({
22-
imports: [CommonModule,GridjsAngularModule],
23-
declarations: [...],
24-
exports: [...],
25-
})
26-
export class AppModule {}
27-
```
28-
2916
In your component template
3017

3118
```ts
3219
import { Component } from '@angular/core';
20+
import { GridjsAngularModule } from 'gridjs-angular';
3321
import { Config } from 'gridjs';
3422

3523
@Component({
24+
standalone: true,
25+
imports: [GridJsAngularComponent],
3626
template: `
3727
<gridjs-angular
3828
[gridConfig]="gridConfig"
@@ -129,4 +119,3 @@ The `gridjs-angular` repository is a monorepo that uses [Nx](https://nx.dev) and
129119
- `pnpm install` - Install all dependencies
130120
- `nx serve demo` - Run demo app
131121
- `nx migrate latest` - Update Nx to the latest version, and upgrade all packages from package.json to their latest version
132-
- `nx update-bindings gridjs-angular` - Update the input and output bindings from GridJS to the Angular component. This command should be run after updating the GridJS version.

packages/gridjs-angular/project.json

+3-1
Original file line numberDiff line numberDiff line change
@@ -34,7 +34,9 @@
3434
},
3535
"update-bindings": {
3636
"executor": "nx:run-commands",
37-
"outputs": ["{workspaceRoot}/packages/gridjs-angular/src/lib/gridjs-binding-base.ts"],
37+
"outputs": [
38+
"{workspaceRoot}/packages/gridjs-angular/src/lib/gridjs-binding-base.ts"
39+
],
3840
"options": {
3941
"command": "node scripts/update-bindings.mjs"
4042
}

packages/gridjs-angular/src/index.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,2 +1,2 @@
11
export * from './lib/gridjs-angular.component';
2-
export { GRID_EVENTS as GRID_JS_EVENTS } from './lib/gridjs-binding-base';
2+
export * from './lib/constants';
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,35 @@
1+
import type { Config } from 'gridjs';
2+
import { GridEvents } from 'gridjs/dist/src/events';
3+
4+
export const GRID_JS_EVENTS: (keyof GridEvents)[] = [
5+
'beforeLoad',
6+
'cellClick',
7+
'load',
8+
'rowClick',
9+
'ready',
10+
];
11+
12+
export const GRID_JS_PROPS: (keyof Config)[] = [
13+
'eventEmitter',
14+
'plugin',
15+
'data',
16+
'server',
17+
'header',
18+
'from',
19+
'storage',
20+
'pipeline',
21+
'autoWidth',
22+
'width',
23+
'height',
24+
'translator',
25+
'style',
26+
'className',
27+
'fixedHeader',
28+
'columns',
29+
'search',
30+
'pagination',
31+
'sort',
32+
'language',
33+
'plugins',
34+
'processingThrottleMs',
35+
];

packages/gridjs-angular/src/lib/gridjs-angular.component.ts

+100-26
Original file line numberDiff line numberDiff line change
@@ -3,73 +3,147 @@ import {
33
Component,
44
ElementRef,
55
EventEmitter,
6+
Input,
7+
OnChanges,
68
OnDestroy,
79
Output,
10+
ViewEncapsulation,
811
} from '@angular/core';
12+
import { CommonModule } from '@angular/common';
913
import { Config, Grid } from 'gridjs';
10-
import { GRID_EVENTS, GridJsAngularBindingBase } from './gridjs-binding-base';
14+
import { GRID_JS_EVENTS, GRID_JS_PROPS } from './constants';
1115
import { GridEvents } from 'gridjs/dist/src/events';
1216

13-
/** only properties that exist on the Config interface (not the Config class) */
14-
type EventName = keyof GridEvents;
15-
type EventHandler = (...args: any[]) => void;
17+
type GridJsAngularComponentProps = Omit<
18+
Partial<Config>,
19+
'instance' | 'store' | 'assign' | 'update'
20+
>;
21+
type GridEvent = keyof GridEvents;
22+
type GridEventData<T extends GridEvent> = Parameters<
23+
GridEvents[T]
24+
>[0] extends undefined
25+
? void
26+
: Parameters<GridEvents[T]>[0];
1627

28+
type AnyFn = (...args: any[]) => any;
1729
@Component({
1830
selector: 'gridjs-angular',
19-
standalone: true,
2031
template: '',
32+
standalone: true,
33+
imports: [CommonModule],
34+
encapsulation: ViewEncapsulation.None,
2135
})
2236
export class GridJsAngularComponent
23-
extends GridJsAngularBindingBase
24-
implements AfterViewInit, OnDestroy
37+
implements AfterViewInit, OnChanges, OnDestroy, GridJsAngularComponentProps
2538
{
26-
private readonly listeners = new Map<EventName, EventHandler>();
39+
private nativeElement: HTMLElement;
40+
private instance?: Grid;
41+
private initialized = false;
42+
private listeners: Map<GridEvent, AnyFn> = new Map();
43+
@Input() config?: Partial<Config>;
2744

28-
/** alias of `load` event due to possible conflict with native load event */
29-
@Output() readonly gridLoad = this.load;
45+
// props
46+
@Input() plugins: Config['plugins'] = [];
47+
@Input() eventEmitter?: Config['eventEmitter'];
48+
@Input() plugin?: Config['plugin'];
49+
@Input() data: Config['data'];
50+
@Input() server: Config['server'];
51+
@Input() header: Config['header'];
52+
@Input() from?: Config['from'];
53+
@Input() storage?: Config['storage'];
54+
@Input() pipeline?: Config['pipeline'];
55+
@Input() autoWidth?: Config['autoWidth'];
56+
@Input() width?: Config['width'];
57+
@Input() height?: Config['height'];
58+
@Input() translator?: Config['translator'];
59+
@Input() style: Config['style'];
60+
@Input() className: Config['className'];
61+
@Input() fixedHeader?: Config['fixedHeader'];
62+
@Input() columns?: Config['columns'];
63+
@Input() search?: Config['search'];
64+
@Input() pagination?: Config['pagination'];
65+
@Input() sort?: Config['sort'];
66+
@Input() language?: Config['language'];
67+
@Input() resizable?: Config['resizable'];
68+
@Input() processingThrottleMs?: Config['processingThrottleMs'];
3069

31-
constructor(private readonly host: ElementRef) {
32-
super();
70+
// events
71+
@Output() beforeLoad: EventEmitter<GridEventData<'beforeLoad'>> =
72+
new EventEmitter(true);
73+
// renamed load event to avoid conflict with native load event
74+
@Output() gridLoad: EventEmitter<GridEventData<'load'>> = new EventEmitter(
75+
true,
76+
);
77+
@Output() cellClick: EventEmitter<GridEventData<'cellClick'>> =
78+
new EventEmitter(true);
79+
@Output() rowClick: EventEmitter<GridEventData<'rowClick'>> =
80+
new EventEmitter(true);
81+
@Output() ready: EventEmitter<GridEventData<'ready'>> = new EventEmitter(
82+
true,
83+
);
84+
85+
constructor(private elementDef: ElementRef) {
86+
this.nativeElement = this.elementDef.nativeElement;
3387
}
3488

3589
ngAfterViewInit(): void {
36-
const instance = new Grid(this.config());
37-
this.instance.set(instance);
90+
this.instance = new Grid(this.getConfig(this.config ?? {}));
3891
this.registerEvents();
39-
instance.render(this.host.nativeElement);
92+
this.instance.render(this.nativeElement);
93+
this.initialized = true;
4094
}
4195

42-
ngOnDestroy(): void {
43-
if (this.instance()) {
44-
this.unregisterEvents();
45-
this.instance.set(undefined);
96+
ngOnChanges(): void {
97+
if (this.initialized) {
98+
this.updateConfig(this.config);
4699
}
47100
}
48101

102+
ngOnDestroy(): void {
103+
if (this.initialized) {
104+
if (this.instance) {
105+
this.unregisterEvents();
106+
this.instance = undefined;
107+
}
108+
}
109+
}
49110
// public api to interact with grid instance
50111
getGridInstance() {
51-
return this.instance();
112+
return this.instance;
52113
}
53114

54115
updateConfig(config: Partial<Config> = {}) {
55-
this.gridConfig.set(config);
116+
this.instance?.updateConfig(this.getConfig(config)).forceRender();
56117
}
57118

58119
private registerEvents() {
59-
for (const event of GRID_EVENTS) {
60-
const emitter = (<any>this)[event] as EventEmitter<any>;
120+
for (const event of GRID_JS_EVENTS) {
121+
const emitter = event === 'load' ? this.gridLoad : this[event];
61122
if (!emitter) {
62123
continue;
63124
}
64-
const listener = (...args: any[]) => emitter.emit(args);
125+
const listener: AnyFn = (args) => emitter.emit(args);
65126
this.listeners.set(event, listener);
66-
this.instance()?.on(event, listener);
127+
if (emitter) {
128+
this.instance?.on(event, listener);
129+
}
67130
}
68131
}
69132

70133
private unregisterEvents() {
71134
for (const [event, listener] of this.listeners.entries()) {
72-
this.instance()?.off(event, listener);
135+
this.instance?.off(event, listener);
136+
}
137+
}
138+
139+
private getConfig(config: Partial<Config> = {}) {
140+
const newConfig: Record<string, unknown> = structuredClone(config);
141+
for (const [key, value] of Object.entries(this)) {
142+
if (GRID_JS_PROPS.includes(key as keyof Config)) {
143+
newConfig[key] = value;
144+
}
73145
}
146+
this.config = newConfig;
147+
return newConfig;
74148
}
75149
}

0 commit comments

Comments
 (0)