HMR не обновляет представление с помощью Angular CLI

Недавно я обновил свое угловое приложение до последних версий. И после ночи ночных кошмаров у меня все заработало, кроме HMR. Я плохо застрял с ним. Ниже приведены мои конфигурации в соответствии с вики HMR Story на Angular CLI:

угловой.json

      "build": {
              "configurations": {
                "hmr": {
                  "fileReplacements": [
                    {
                      "replace": "src/environments/environment.ts",
                      "with": "src/environments/environment.hmr.ts"
                    }
                  ]
                }
              }
            },
 "serve": {
          "configurations": {
            "hmr": {
              "hmr": true,
              "browserTarget": "appHit:build:hmr"
            },
          }
        },

hmr.js

import { NgModuleRef, ApplicationRef } from '@angular/core';
import { createNewHosts } from '@angularclass/hmr';

export const hmrBootstrap = (module: any, bootstrap: () => Promise<NgModuleRef<any>>) => {
  let ngModule: NgModuleRef<any>;
  module.hot.accept();
  bootstrap().then(mod => ngModule = mod);
  module.hot.dispose(() => {
    const appRef: ApplicationRef = ngModule.injector.get(ApplicationRef);
    const elements = appRef.components.map(c => c.location.nativeElement);
    const makeVisible = createNewHosts(elements);
    ngModule.destroy();
    makeVisible();
  });
};

main.ts

import { enableProdMode } from '@angular/core';
import { platformBrowserDynamic } from '@angular/platform-browser-dynamic';

import { AppModule } from './app/app.module';
import { environment } from './environments/environment';

import { hmrBootstrap } from './hmr';

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

const bootstrap = () => platformBrowserDynamic().bootstrapModule(AppModule);

if (environment.hmr) {
  if (module[ 'hot' ]) {
    hmrBootstrap(module, bootstrap);
  } else {
    console.error('HMR is not enabled for webpack-dev-server!');
    console.log('Are you using the --hmr flag for ng serve?');
  }
} else {
    bootstrap().catch(err => console.log(err));
}

Я пробовал следующие команды:

ng serve --hmr

ng serve --hmr --configuration hmr

ng serve --configuration hmr

Все компилируется при изменении, и даже запущенные события кэшируются в браузере, но ничего не происходит после того, как HMR регистрирует следующее:

[WDS] App updated. Recompiling...
[WDS] App hot update...

Я полностью потерян в этот момент. Любая помощь будет очень признательна. Спасибо


person Metabolic    schedule 12.06.2018    source источник
comment
Все еще проблема в связи с route-resolve, см. здесь: stackoverflow.com/questions/55355133/   -  person Lonely    schedule 26.03.2019


Ответы (4)


Мы столкнулись с той же проблемой. Это было решено путем удаления зависимостей разработчика, связанных с Webpack, и выполнения новой установки npm.

rm -R package-lock.json node_modules npm cache clean --force npm i

Надеюсь, это поможет.

person samuel rott    schedule 25.07.2018
comment
У меня была такая же проблема. Раньше я пытался удалить свои node_modules и переустановить их, но я думаю, что ключ был в том, что у нас все еще были некоторые старые зависимости от разработчиков веб-пакетов в нашем package.json со времен до CLI. - person wags1999; 09.10.2018
comment
Удалите package-lock.json перед переустановкой пакетов! Дох! - person The Head Rush; 04.09.2019
comment
Действительно, мы перешли с веб-пакета на angular cli, и некоторые зависимости разработчиков все еще остались в package.json. - person stephan.peters; 10.02.2020

Немного поздно для этого ... то, что сработало для меня, было в моей папке ClientApp. У меня была папка / dist со всеми битами js ... HMR не будет работать, если есть папка dist. Dist попадает туда, если вы выполняете сборку. Удалите папку dist, затем запустите ng serve. В моем случае у меня есть основное приложение .net, на котором оно размещено, поэтому я создал его и запустил для своего углового приложения... протестировал, изменив html-файл... приложение должно перезагрузиться в браузере с новым содержимым.

person swandog    schedule 28.09.2018
comment
Для меня наличие папки dist не имело значения. Проблема заключалась в некоторых старых зависимостях разработчика веб-пакета, как это было предложено Сэмюэлем Роттом. - person wags1999; 09.10.2018

Я думаю, что это может быть полезно для некоторых. Я решил свою проблему, обновив Angular до версии 7 и добавив следующую строку в main.ts для HMR.

module['hot'].accept();

Следующим образом:

if (environment.hmr) {
    if (module['hot']) {
        module['hot'].accept();
        hmrBootstrap(module, bootstrap);
    } else {
        console.error('HMR is not enabled for webpack-dev-server!');
        console.log('Are you using the --hmr flag for ng serve?');
    }
} else {
    console.log('hot');
    bootstrap().catch(err => console.log(err));
}

До сих пор HMR работал совершенно нормально. У меня не было времени на дальнейшую отладку, но, скорее всего, это была несовместимость зависимостей, которая могла вызвать это с моей стороны.

person Metabolic    schedule 11.04.2019
comment
Не работает для меня. На лицо есть строка module.hot.accept(); в функции hmrBootstrap, которая, похоже, делает то же самое. - person Walter Luszczyk; 20.11.2019
comment
Это не могло решить вашу проблему, возможно, что-то еще. Эта строка кода уже существует в hmrBootstrap, как предложил @WalterLuszczyk. Так что повторный вызов ничего не изменит. - person Pankaj; 17.01.2020

Вот моя установка, в настоящее время она отлично работает в новейших версиях. Вы можете избавиться от ngrx вещей, если они вам не нужны :-)

// 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

// ...
export class AppModule {
  constructor(
    public appRef: ApplicationRef 
    // , ...
  ){}
}
// angular.json

"build": {
  "configurations": {
    "hmr": {
      "fileReplacements": [
        {
          "replace": "src/environments/environment.ts",
          "with": "src/environments/environment.hmr.ts"
        }
      ]
    }
  }
},
"serve": {
  "configurations": {
    "hmr": {
      "browserTarget": "AppName:build:hmr"
    }
  }
}

И я запускаю его с ng serve --hmr -c=hmr

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

person Heehaaw    schedule 14.06.2018