Горячая замена модуля перезагружает все приложение вместо конкретного компонента - PullRequest
0 голосов
/ 13 сентября 2018

Я создал новый угловой проект и настроил HMR, как описано здесь: https://github.com/angular/angular-cli/wiki/stories-configure-hmr

Проект содержит основной компонент (родительский), который имеет router-outlet и ссылки на 3 дочерних компонента, которые загружаются с отложенной загрузкой.

Примечание: я также использую пользовательский RouteReuseStrategy, но он не влияет на HMR, насколько я тестировал.

Независимо от того, какой файл я изменяю - .html или .ts (parent / children), все приложение перезагружается.

Я установил базовое репо, которое можно найти здесь: https://github.com/ronfogel/demo-hmr

Ответы [ 2 ]

0 голосов
/ 09 октября 2018

Такое поведение ожидается, я попытаюсь объяснить, что происходит.

Горячая замена модуля, которую настроил angular, на самом деле просто перезагружает все приложение более общим способом и с поддержкой несколькихкорни приложения, но если вы отложите абстракции в сторону, то просто удалите тег app-root, добавите его снова и снова загрузите AppModule, поэтому все приложение изменится:

export const hmrBootstrap = (
  // webpack stuff
  module: any,
  // bootstrap is AppModule bootstrapper 
  bootstrap: () => Promise<NgModuleRef<any>>
) => {
  let ngModule: NgModuleRef<any>;
  module.hot.accept();
  // bootstraps AppModule ecery time a HMR is needed
  // sets ngModule equal to AppModule if successful (unnecessary)
  bootstrap().then(mod => (ngModule = mod));
  module.hot.dispose(() => {
    // next two lines get native element for all `app-root` tags
    // that exist in `index.html`
    const appRef: ApplicationRef = ngModule.injector.get(ApplicationRef);
    const elements = appRef.components.map(c => c.location.nativeElement);
    // I will share createNewHosts code below it's nothing fancy just
    // the simple add and delete i mentioned
    const makeVisible = createNewHosts(elements);
    //destroy the current AppModule and finalize deletion
    ngModule.destroy();
    makeVisible();
  });
};
0 голосов
/ 17 сентября 2018

Это то, что я использую для последней версии Angular, просто отлично работает. Вы можете попробовать ...

// main.ts
import { bootloader, createInputTransfer, createNewHosts, removeNgStyles } 
    from '@angularclass/hmr/dist/helpers'; // For correct treeshaking

if (environment.production) {
  enableProdMode();
}

type HmrModule<S> = { appRef: ApplicationRef }
type HmrNgrxModule<S, A> = HmrModule<S> & { 
  store: { dispatch: (A) => any } & Observable<S>,
  actionCreator: (s: S) => A
}

const isNgrxModule = <S, A, M extends HmrNgrxModule<S, A>>
  (instance: HmrModule<S> | HmrNgrxModule<S, A>): instance is M =>
    !!((<M>instance).store && (<M>instance).actionCreator);

function processModule<S, A, M extends HmrModule<S> | HmrNgrxModule<S, A>>(ngModuleRef: NgModuleRef<M>) {

  const hot = module['hot'];
  if (hot) {

    hot['accept']();

    const instance = ngModuleRef.instance;
    const hmrStore = hot['data'];

    if (hmrStore) {
      hmrStore.rootState 
        && isNgrxModule(instance) 
        && instance.store.dispatch(instance.actionCreator(hmrStore.rootState));
      hmrStore.restoreInputValues && hmrStore.restoreInputValues();
      instance.appRef.tick();
      Object.keys(hmrStore).forEach(prop => delete hmrStore[prop]);
    }

    hot['dispose'](hmrStore => {
      isNgrxModule(instance) && instance.store.pipe(take(1)).subscribe(s => hmrStore.rootState = s);
      const cmpLocation = instance.appRef.components.map(cmp => cmp.location.nativeElement);
      const disposeOldHosts = createNewHosts(cmpLocation);
      hmrStore.restoreInputValues = createInputTransfer();
      removeNgStyles();
      ngModuleRef.destroy();
      disposeOldHosts();
    });
  }
  else {
    console.error('HMR is not enabled for webpack-dev-server!');
    console.log('Are you using the --hmr flag for ng serve?');
  }

  return ngModuleRef;
}

const bootstrap = () => platformBrowserDynamic().bootstrapModule(AppModule);
const hmrBootstrap = () => bootloader(() => bootstrap().then(processModule));

environment.hmr
  ? hmrBootstrap()
  : bootstrap();
// app.module.ts
@NgModule({ ... })
export class AppModule {
  constructor(public appRef: ApplicationRef) { ... }
}

Настройка HMR будет работать и с хранилищем Ngrx, если вы занимаетесь этим. Вы можете не указывать код обработки Ngrx.

Надеюсь, это немного поможет: -)

...