Angular Universal:导航器未定义

问题描述 投票:0回答:4

我按照官方 Angular-CLI 教程 将 Angular-Universal 集成到我现有的 Angular-CLI 应用程序中。

我能够为我的 angular-cli 应用程序执行 SSR。但是当我尝试集成 ngx-leaflet 时,我收到以下错误:

ReferenceError:导航器未定义 在D: g2-ssr-pwa\dist\server.js:40251:29

现在,我了解到传单正在尝试访问在节点上下文中不可用的导航器对象。因此,我决定延迟传单渲染,直到页面加载到浏览器中,如该 SO thread 中所示。 但我仍然遇到同样的错误。您可以在这里查看带有传单问题的演示应用程序。

./src/app/browserModuleLoader.service.ts:

import { Component, Inject, Injectable, OnInit, PLATFORM_ID } from '@angular/core';
import { isPlatformBrowser, isPlatformServer } from '@angular/common';

@Injectable()
export class BrowserModuleLoaderService {
    private _L: any;

    public constructor(@Inject(PLATFORM_ID) private _platformId: Object) {
        this._init();
    }

    public getL() {
        return this._safeGet(() => this._L);
    }

    private _init() {
        if (isPlatformBrowser(this._platformId)) {
            this._requireLegacyResources();
        }
    }

    private _requireLegacyResources() {
        this._L = require('leaflet');
    }

    private _safeGet(getCallcack: () => any) {
        if (isPlatformServer(this._platformId)) {
            throw new Error('invalid access to legacy component on server');
        }

        return getCallcack();
    }
}

./src/app/leaflet/app/leaflet.component.ts:

// import * as L from 'leaflet';

import { ChangeDetectionStrategy, ChangeDetectorRef, Component, Inject, PLATFORM_ID } from '@angular/core';

import { BrowserModuleLoaderService } from '../browserModuleLoader.service';
import { isPlatformBrowser } from '@angular/common';

@Component({
    selector: 'app-leaflet',
    styleUrls: ['./leaflet.component.scss'],
    template: `
      <div  *ngIf="isBrowser">
        <div leaflet [leafletOptions]="options"></div>
        </div>
  `,
    changeDetection: ChangeDetectionStrategy.OnPush,
})
export class LeafletComponent {
    isBrowser: boolean;
    options = {};

    constructor(private cdr: ChangeDetectorRef,
        @Inject(PLATFORM_ID) platformId: Object,
        private browserModuleLoaderService: BrowserModuleLoaderService
    ) {
        this.isBrowser = isPlatformBrowser(platformId);
    }

    ngAfterViewInit() {
        console.log('this.isBrowser ', this.isBrowser);
        if (this.isBrowser) {
            const L = this.browserModuleLoaderService.getL();
            this.options = {
                layers: [
                    L.tileLayer('http://{s}.tile.openstreetmap.org/{z}/{x}/{y}.png', { maxZoom: 18, attribution: '...' }),
                ],
                zoom: 5,
                center: L.latLng({ lat: 38.991709, lng: -76.886109 }),
            };
        }
        this.cdr.detach();
    }

}

./src/app/app.component.html:

<div>
  <app-leaflet></app-leaflet>
</div>

如何安全地延迟传单渲染,直到平台不是浏览器?

编辑:

我删除了与传单相关的所有代码(browserModuleLoader.service.ts、leaflet.component.ts 等),并仅在 app.module.ts 中保留传单模块导入,实际上此导入导致了问题。

./src/app/app.module.ts:

import { AppComponent } from './app.component';
import { BrowserModule } from '@angular/platform-browser';
// import { BrowserModuleLoaderService } from './browserModuleLoader.service';
// import { LeafletComponent } from './leaflet/leaflet.component';
import { LeafletModule } from '@asymmetrik/ngx-leaflet';
import { NgModule } from '@angular/core';

@NgModule({
  declarations: [
    AppComponent,
    // LeafletComponent
  ],
  imports: [
    BrowserModule.withServerTransition({appId: 'my-app'}),
    LeafletModule.forRoot()
  ],
  providers: [
    // BrowserModuleLoaderService
  ],
  bootstrap: [AppComponent]
})
export class AppModule { }

./src/app/app.server.module.ts:

import {AppComponent} from './app.component';
import {AppModule} from './app.module';
import {ModuleMapLoaderModule} from '@nguniversal/module-map-ngfactory-loader';
import {NgModule} from '@angular/core';
import {ServerModule} from '@angular/platform-server';

@NgModule({
  imports: [
    AppModule,
    ServerModule,
    ModuleMapLoaderModule
  ],
  bootstrap: [AppComponent],
})
export class AppServerModule {}

如何处理此 nxg-leaflet 模块导入?

javascript angular angular-cli angular-universal ngx-leaflet
4个回答
12
投票

使用Mock Browser解决了这个问题。

服务器.ts:

const MockBrowser = require('mock-browser').mocks.MockBrowser;
const mock = new MockBrowser();


global['navigator'] = mock.getNavigator();

0
投票

我在使用 Angular Universal 时也收到了这个错误。使用上面解决方案中的模拟浏览器确实为我解决了这个错误,但它也启动了一个与 CommonJS 相关的新警告。我没有深入研究这个问题,而是意识到我已经在 server.ts 文件中使用了 Domino,因此我可以轻松地使用 Domino 设置导航器。这对我来说最有效:

npm install domino

服务器.ts:

const domino = require('domino');
const fs = require('fs');
const path = require('path');
const template = fs
  .readFileSync(path.join('dist/<your-app-name>/browser', 'index.html')) //<--- REPLACE WITH YOUR APP NAME
  .toString();
const window = domino.createWindow(template);
global['window'] = window;
global['document'] = window.document;
global['navigator'] = window.navigator;

0
投票

@2024 角度 17
我收到了 Mock Browser 使用已弃用组件的警告,经过一番研究,我发现 Navigator 是一个窗口对象,因此如果您打算使用浏览器,可以添加服务器检查,请看一下:

import { isPlatformBrowser } from '@angular/common';
import { Inject, PLATFORM_ID } from '@angular/core';

export class AppComponent {

  constructor(
    @Inject(PLATFORM_ID) private platformId: Object
  ) {
    if (isPlatformBrowser(this.platformId)) {
      // Use Navigator
    }
  }
}

-1
投票

(global as any).navigator = win.navigator;
修复,更加优雅,绝对是原生的,并且不依赖过时的模拟浏览器

© www.soinside.com 2019 - 2024. All rights reserved.