@angular/pwa 17.0.0-next.3 → 17.0.0-next.5

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/README.md CHANGED
@@ -1,22 +1,23 @@
1
1
  # `@angular/pwa`
2
2
 
3
3
  This is a [schematic](https://angular.io/guide/schematics) for adding
4
- [Progress Web App](https://web.dev/progressive-web-apps/) support to an Angular app. Run the
4
+ [Progressive Web App](https://web.dev/progressive-web-apps/) support to an Angular project. Run the
5
5
  schematic with the [Angular CLI](https://angular.io/cli):
6
6
 
7
7
  ```shell
8
- ng add @angular/pwa
8
+ ng add @angular/pwa --project <project-name>
9
9
  ```
10
10
 
11
- This makes a few changes to your project:
11
+ Executing the command mentioned above will perform the following actions:
12
12
 
13
- 1. Adds [`@angular/service-worker`](https://npmjs.com/@angular/service-worker) as a dependency.
13
+ 1. Adds [`@angular/service-worker`](https://npmjs.com/@angular/service-worker) as a dependency to your project.
14
14
  1. Enables service worker builds in the Angular CLI.
15
- 1. Imports and registers the service worker in the app module.
16
- 1. Adds a [web app manifest](https://developer.mozilla.org/en-US/docs/Web/Manifest).
17
- 1. Updates the `index.html` file to link to the manifest and set theme colors.
18
- 1. Adds required icons for the manifest.
19
- 1. Creates a config file `ngsw-config.json`, specifying caching behaviors and other settings.
15
+ 1. Imports and registers the service worker in the application module.
16
+ 1. Updates the `index.html` file:
17
+ - Includes a link to add the [manifest.webmanifest](https://developer.mozilla.org/en-US/docs/Web/Manifest) file.
18
+ - Adds a meta tag for `theme-color`.
19
+ 1. Installs icon files to support the installed Progressive Web App (PWA).
20
+ 1. Creates the service worker configuration file called `ngsw-config.json`, specifying caching behaviors and other settings.
20
21
 
21
22
  See [Getting started with service workers](https://angular.io/guide/service-worker-getting-started)
22
23
  for more information.
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@angular/pwa",
3
- "version": "17.0.0-next.3",
3
+ "version": "17.0.0-next.5",
4
4
  "description": "PWA schematics for Angular",
5
5
  "keywords": [
6
6
  "Angular CLI",
@@ -17,8 +17,8 @@
17
17
  "save": false
18
18
  },
19
19
  "dependencies": {
20
- "@angular-devkit/schematics": "17.0.0-next.3",
21
- "@schematics/angular": "17.0.0-next.3",
20
+ "@angular-devkit/schematics": "17.0.0-next.5",
21
+ "@schematics/angular": "17.0.0-next.5",
22
22
  "parse5-html-rewriting-stream": "7.0.0"
23
23
  },
24
24
  "peerDependencies": {
package/pwa/index.js CHANGED
@@ -84,7 +84,8 @@ function default_1(options) {
84
84
  const buildTargets = [];
85
85
  const testTargets = [];
86
86
  for (const target of project.targets.values()) {
87
- if (target.builder === '@angular-devkit/build-angular:browser') {
87
+ if (target.builder === '@angular-devkit/build-angular:browser' ||
88
+ target.builder === '@angular-devkit/build-angular:application') {
88
89
  buildTargets.push(target);
89
90
  }
90
91
  else if (target.builder === '@angular-devkit/build-angular:karma') {
@@ -150,4 +151,4 @@ exports.default = default_1;
150
151
  function loadEsmModule(modulePath) {
151
152
  return new Function('modulePath', `return import(modulePath);`)(modulePath);
152
153
  }
153
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../../../../../../../packages/angular/pwa/pwa/index.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\n\nimport {\n  Rule,\n  SchematicsException,\n  Tree,\n  apply,\n  chain,\n  externalSchematic,\n  mergeWith,\n  move,\n  template,\n  url,\n} from '@angular-devkit/schematics';\nimport { readWorkspace, writeWorkspace } from '@schematics/angular/utility';\nimport { posix } from 'path';\nimport { Readable, Writable } from 'stream';\nimport { Schema as PwaOptions } from './schema';\n\nfunction updateIndexFile(path: string): Rule {\n  return async (host: Tree) => {\n    const buffer = host.read(path);\n    if (buffer === null) {\n      throw new SchematicsException(`Could not read index file: ${path}`);\n    }\n\n    const { RewritingStream } = await loadEsmModule<typeof import('parse5-html-rewriting-stream')>(\n      'parse5-html-rewriting-stream',\n    );\n\n    const rewriter = new RewritingStream();\n    let needsNoScript = true;\n    rewriter.on('startTag', (startTag) => {\n      if (startTag.tagName === 'noscript') {\n        needsNoScript = false;\n      }\n\n      rewriter.emitStartTag(startTag);\n    });\n\n    rewriter.on('endTag', (endTag) => {\n      if (endTag.tagName === 'head') {\n        rewriter.emitRaw('  <link rel=\"manifest\" href=\"manifest.webmanifest\">\\n');\n        rewriter.emitRaw('  <meta name=\"theme-color\" content=\"#1976d2\">\\n');\n      } else if (endTag.tagName === 'body' && needsNoScript) {\n        rewriter.emitRaw(\n          '  <noscript>Please enable JavaScript to continue using this application.</noscript>\\n',\n        );\n      }\n\n      rewriter.emitEndTag(endTag);\n    });\n\n    return new Promise<void>((resolve) => {\n      const input = new Readable({\n        encoding: 'utf8',\n        read(): void {\n          this.push(buffer);\n          this.push(null);\n        },\n      });\n\n      const chunks: Array<Buffer> = [];\n      const output = new Writable({\n        write(chunk: string | Buffer, encoding: BufferEncoding, callback: Function): void {\n          chunks.push(typeof chunk === 'string' ? Buffer.from(chunk, encoding) : chunk);\n          callback();\n        },\n        final(callback: (error?: Error) => void): void {\n          const full = Buffer.concat(chunks);\n          host.overwrite(path, full.toString());\n          callback();\n          resolve();\n        },\n      });\n\n      input.pipe(rewriter).pipe(output);\n    });\n  };\n}\n\nexport default function (options: PwaOptions): Rule {\n  return async (host) => {\n    if (!options.title) {\n      options.title = options.project;\n    }\n\n    const workspace = await readWorkspace(host);\n\n    if (!options.project) {\n      throw new SchematicsException('Option \"project\" is required.');\n    }\n\n    const project = workspace.projects.get(options.project);\n    if (!project) {\n      throw new SchematicsException(`Project is not defined in this workspace.`);\n    }\n\n    if (project.extensions['projectType'] !== 'application') {\n      throw new SchematicsException(`PWA requires a project type of \"application\".`);\n    }\n\n    // Find all the relevant targets for the project\n    if (project.targets.size === 0) {\n      throw new SchematicsException(`Targets are not defined for this project.`);\n    }\n\n    const buildTargets = [];\n    const testTargets = [];\n    for (const target of project.targets.values()) {\n      if (target.builder === '@angular-devkit/build-angular:browser') {\n        buildTargets.push(target);\n      } else if (target.builder === '@angular-devkit/build-angular:karma') {\n        testTargets.push(target);\n      }\n    }\n\n    // Add manifest to asset configuration\n    const assetEntry = posix.join(\n      project.sourceRoot ?? posix.join(project.root, 'src'),\n      'manifest.webmanifest',\n    );\n    for (const target of [...buildTargets, ...testTargets]) {\n      if (target.options) {\n        if (Array.isArray(target.options.assets)) {\n          target.options.assets.push(assetEntry);\n        } else {\n          target.options.assets = [assetEntry];\n        }\n      } else {\n        target.options = { assets: [assetEntry] };\n      }\n    }\n\n    // Find all index.html files in build targets\n    const indexFiles = new Set<string>();\n    for (const target of buildTargets) {\n      if (typeof target.options?.index === 'string') {\n        indexFiles.add(target.options.index);\n      }\n\n      if (!target.configurations) {\n        continue;\n      }\n\n      for (const options of Object.values(target.configurations)) {\n        if (typeof options?.index === 'string') {\n          indexFiles.add(options.index);\n        }\n      }\n    }\n\n    // Setup sources for the assets files to add to the project\n    const sourcePath = project.sourceRoot ?? posix.join(project.root, 'src');\n\n    // Setup service worker schematic options\n    const { title, ...swOptions } = options;\n\n    await writeWorkspace(host, workspace);\n\n    return chain([\n      externalSchematic('@schematics/angular', 'service-worker', swOptions),\n      mergeWith(apply(url('./files/root'), [template({ ...options }), move(sourcePath)])),\n      mergeWith(apply(url('./files/assets'), [move(posix.join(sourcePath, 'assets'))])),\n      ...[...indexFiles].map((path) => updateIndexFile(path)),\n    ]);\n  };\n}\n\n/**\n * This uses a dynamic import to load a module which may be ESM.\n * CommonJS code can load ESM code via a dynamic import. Unfortunately, TypeScript\n * will currently, unconditionally downlevel dynamic import into a require call.\n * require calls cannot load ESM code and will result in a runtime error. To workaround\n * this, a Function constructor is used to prevent TypeScript from changing the dynamic import.\n * Once TypeScript provides support for keeping the dynamic import this workaround can\n * be dropped.\n *\n * @param modulePath The path of the module to load.\n * @returns A Promise that resolves to the dynamically imported module.\n */\nfunction loadEsmModule<T>(modulePath: string | URL): Promise<T> {\n  return new Function('modulePath', `return import(modulePath);`)(modulePath) as Promise<T>;\n}\n"]}
154
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../../../../../../../packages/angular/pwa/pwa/index.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\n\nimport {\n  Rule,\n  SchematicsException,\n  Tree,\n  apply,\n  chain,\n  externalSchematic,\n  mergeWith,\n  move,\n  template,\n  url,\n} from '@angular-devkit/schematics';\nimport { readWorkspace, writeWorkspace } from '@schematics/angular/utility';\nimport { posix } from 'path';\nimport { Readable, Writable } from 'stream';\nimport { Schema as PwaOptions } from './schema';\n\nfunction updateIndexFile(path: string): Rule {\n  return async (host: Tree) => {\n    const buffer = host.read(path);\n    if (buffer === null) {\n      throw new SchematicsException(`Could not read index file: ${path}`);\n    }\n\n    const { RewritingStream } = await loadEsmModule<typeof import('parse5-html-rewriting-stream')>(\n      'parse5-html-rewriting-stream',\n    );\n\n    const rewriter = new RewritingStream();\n    let needsNoScript = true;\n    rewriter.on('startTag', (startTag) => {\n      if (startTag.tagName === 'noscript') {\n        needsNoScript = false;\n      }\n\n      rewriter.emitStartTag(startTag);\n    });\n\n    rewriter.on('endTag', (endTag) => {\n      if (endTag.tagName === 'head') {\n        rewriter.emitRaw('  <link rel=\"manifest\" href=\"manifest.webmanifest\">\\n');\n        rewriter.emitRaw('  <meta name=\"theme-color\" content=\"#1976d2\">\\n');\n      } else if (endTag.tagName === 'body' && needsNoScript) {\n        rewriter.emitRaw(\n          '  <noscript>Please enable JavaScript to continue using this application.</noscript>\\n',\n        );\n      }\n\n      rewriter.emitEndTag(endTag);\n    });\n\n    return new Promise<void>((resolve) => {\n      const input = new Readable({\n        encoding: 'utf8',\n        read(): void {\n          this.push(buffer);\n          this.push(null);\n        },\n      });\n\n      const chunks: Array<Buffer> = [];\n      const output = new Writable({\n        write(chunk: string | Buffer, encoding: BufferEncoding, callback: Function): void {\n          chunks.push(typeof chunk === 'string' ? Buffer.from(chunk, encoding) : chunk);\n          callback();\n        },\n        final(callback: (error?: Error) => void): void {\n          const full = Buffer.concat(chunks);\n          host.overwrite(path, full.toString());\n          callback();\n          resolve();\n        },\n      });\n\n      input.pipe(rewriter).pipe(output);\n    });\n  };\n}\n\nexport default function (options: PwaOptions): Rule {\n  return async (host) => {\n    if (!options.title) {\n      options.title = options.project;\n    }\n\n    const workspace = await readWorkspace(host);\n\n    if (!options.project) {\n      throw new SchematicsException('Option \"project\" is required.');\n    }\n\n    const project = workspace.projects.get(options.project);\n    if (!project) {\n      throw new SchematicsException(`Project is not defined in this workspace.`);\n    }\n\n    if (project.extensions['projectType'] !== 'application') {\n      throw new SchematicsException(`PWA requires a project type of \"application\".`);\n    }\n\n    // Find all the relevant targets for the project\n    if (project.targets.size === 0) {\n      throw new SchematicsException(`Targets are not defined for this project.`);\n    }\n\n    const buildTargets = [];\n    const testTargets = [];\n    for (const target of project.targets.values()) {\n      if (\n        target.builder === '@angular-devkit/build-angular:browser' ||\n        target.builder === '@angular-devkit/build-angular:application'\n      ) {\n        buildTargets.push(target);\n      } else if (target.builder === '@angular-devkit/build-angular:karma') {\n        testTargets.push(target);\n      }\n    }\n\n    // Add manifest to asset configuration\n    const assetEntry = posix.join(\n      project.sourceRoot ?? posix.join(project.root, 'src'),\n      'manifest.webmanifest',\n    );\n    for (const target of [...buildTargets, ...testTargets]) {\n      if (target.options) {\n        if (Array.isArray(target.options.assets)) {\n          target.options.assets.push(assetEntry);\n        } else {\n          target.options.assets = [assetEntry];\n        }\n      } else {\n        target.options = { assets: [assetEntry] };\n      }\n    }\n\n    // Find all index.html files in build targets\n    const indexFiles = new Set<string>();\n    for (const target of buildTargets) {\n      if (typeof target.options?.index === 'string') {\n        indexFiles.add(target.options.index);\n      }\n\n      if (!target.configurations) {\n        continue;\n      }\n\n      for (const options of Object.values(target.configurations)) {\n        if (typeof options?.index === 'string') {\n          indexFiles.add(options.index);\n        }\n      }\n    }\n\n    // Setup sources for the assets files to add to the project\n    const sourcePath = project.sourceRoot ?? posix.join(project.root, 'src');\n\n    // Setup service worker schematic options\n    const { title, ...swOptions } = options;\n\n    await writeWorkspace(host, workspace);\n\n    return chain([\n      externalSchematic('@schematics/angular', 'service-worker', swOptions),\n      mergeWith(apply(url('./files/root'), [template({ ...options }), move(sourcePath)])),\n      mergeWith(apply(url('./files/assets'), [move(posix.join(sourcePath, 'assets'))])),\n      ...[...indexFiles].map((path) => updateIndexFile(path)),\n    ]);\n  };\n}\n\n/**\n * This uses a dynamic import to load a module which may be ESM.\n * CommonJS code can load ESM code via a dynamic import. Unfortunately, TypeScript\n * will currently, unconditionally downlevel dynamic import into a require call.\n * require calls cannot load ESM code and will result in a runtime error. To workaround\n * this, a Function constructor is used to prevent TypeScript from changing the dynamic import.\n * Once TypeScript provides support for keeping the dynamic import this workaround can\n * be dropped.\n *\n * @param modulePath The path of the module to load.\n * @returns A Promise that resolves to the dynamically imported module.\n */\nfunction loadEsmModule<T>(modulePath: string | URL): Promise<T> {\n  return new Function('modulePath', `return import(modulePath);`)(modulePath) as Promise<T>;\n}\n"]}