ngx-bootstrap - Can't resolve all parameters for BsModalService in unit tests

1.7k Views Asked by At

I am using ngx-bootstrap's (3.0.1) ModalService to show a modal dialog containing a component

this.modalRef = this.modalService.show(MyDialogComponent);

and I have this in my unit test (run by jest)

beforeEach(async(() => {
  TestBed.configureTestingModule({
    declarations: [MyComponent, MyDialogComponent],
    schemas: [CUSTOM_ELEMENTS_SCHEMA],
    providers: [MyService],
    imports: [
      ModalModule.forRoot(),
    ],
  })
  .compileComponents();
}));

The stubborn error tells me Can't resolve all parameters for BsModalService: (?, ?). The answers I was able to find were mentioning lack of a proper @Injectable() decorator, but this is obviously beyond my control and besides all it is irrelevant and misleading - the decorator is present in the source code of ngx-bootstrap.

I tried to explicitly provide all the components and services from the package (rather than ModuleWithProviders, but this didn't help either.

Running short of ideas.

Update 1

Tried commenting in and out

import 'core-js/es7/reflect';

in plyfills.ts. Doesn't seem to matter. Unless there is a Jest config setting associated with that that I am missing?

2

There are 2 best solutions below

0
On

As the service from BsModalService dynamically populates the DOM (entryComponents) you can take use of BrowserDynamicTestingModule API by adding into the configureTestingModule:

beforeEach(async(() => {
TestBed.configureTestingModule({
    imports: [ ModalModule.forRoot() ],
    declarations: [ MyDialogComponent ],
}).overrideModule(BrowserDynamicTestingModule, {
  set: {
    entryComponents: [ HelloComponent ],
  }
}).compileComponents();

}));

trigger template elements(e.g. a button) with the dispatch API and then just check if it pops up in the DOM.

here's a stackblitz with a live example.

NOTE: As the ngx BsModalService populates the DOM as a sibling to the root node of angular i was not able to use the Angular API to select the modal element so instead i had to use DOM querySelector. If someone has a better way please comment so i can update

1
On

try to import core-js/es7/reflect in your polyfills.ts, it's needed to make your test work

import 'core-js/es7/reflect';