/** * @license * Copyright Google LLC All Rights Reserved. * * Use of this source code is governed by an MIT-style license that can be * found in the LICENSE file at https://angular.io/license */ import * as html from '../ml_parser/ast'; import { ParseError } from '../parse_util'; import * as t from './r3_ast'; import { getTriggerParametersStart, parseDeferredTime, parseOnTrigger, parseWhenTrigger } from './r3_deferred_triggers'; /** Pattern to identify a `prefetch when` trigger. */ const PREFETCH_WHEN_PATTERN = /^prefetch\s+when\s/; /** Pattern to identify a `prefetch on` trigger. */ const PREFETCH_ON_PATTERN = /^prefetch\s+on\s/; /** Pattern to identify a `minimum` parameter in a block. */ const MINIMUM_PARAMETER_PATTERN = /^minimum\s/; /** Pattern to identify a `after` parameter in a block. */ const AFTER_PARAMETER_PATTERN = /^after\s/; /** Pattern to identify a `when` parameter in a block. */ const WHEN_PARAMETER_PATTERN = /^when\s/; /** Pattern to identify a `on` parameter in a block. */ const ON_PARAMETER_PATTERN = /^on\s/; /** Possible types of secondary deferred blocks. */ export var SecondaryDeferredBlockType; (function (SecondaryDeferredBlockType) { SecondaryDeferredBlockType["PLACEHOLDER"] = "placeholder"; SecondaryDeferredBlockType["LOADING"] = "loading"; SecondaryDeferredBlockType["ERROR"] = "error"; })(SecondaryDeferredBlockType || (SecondaryDeferredBlockType = {})); /** Creates a deferred block from an HTML AST node. */ export function createDeferredBlock(ast, visitor, bindingParser) { const errors = []; const [primaryBlock, ...secondaryBlocks] = ast.blocks; const { triggers, prefetchTriggers } = parsePrimaryTriggers(primaryBlock.parameters, bindingParser, errors); const { placeholder, loading, error } = parseSecondaryBlocks(secondaryBlocks, errors, visitor); return { node: new t.DeferredBlock(html.visitAll(visitor, primaryBlock.children), triggers, prefetchTriggers, placeholder, loading, error, ast.sourceSpan, ast.startSourceSpan, ast.endSourceSpan), errors, }; } function parseSecondaryBlocks(blocks, errors, visitor) { let placeholder = null; let loading = null; let error = null; for (const block of blocks) { try { switch (block.name) { case SecondaryDeferredBlockType.PLACEHOLDER: if (placeholder !== null) { errors.push(new ParseError(block.startSourceSpan, `"defer" block can only have one "${SecondaryDeferredBlockType.PLACEHOLDER}" block`)); } else { placeholder = parsePlaceholderBlock(block, visitor); } break; case SecondaryDeferredBlockType.LOADING: if (loading !== null) { errors.push(new ParseError(block.startSourceSpan, `"defer" block can only have one "${SecondaryDeferredBlockType.LOADING}" block`)); } else { loading = parseLoadingBlock(block, visitor); } break; case SecondaryDeferredBlockType.ERROR: if (error !== null) { errors.push(new ParseError(block.startSourceSpan, `"defer" block can only have one "${SecondaryDeferredBlockType.ERROR}" block`)); } else { error = parseErrorBlock(block, visitor); } break; default: errors.push(new ParseError(block.startSourceSpan, `Unrecognized block "${block.name}"`)); break; } } catch (e) { errors.push(new ParseError(block.startSourceSpan, e.message)); } } return { placeholder, loading, error }; } function parsePlaceholderBlock(ast, visitor) { let minimumTime = null; for (const param of ast.parameters) { if (MINIMUM_PARAMETER_PATTERN.test(param.expression)) { const parsedTime = parseDeferredTime(param.expression.slice(getTriggerParametersStart(param.expression))); if (parsedTime === null) { throw new Error(`Could not parse time value of parameter "minimum"`); } minimumTime = parsedTime; } else { throw new Error(`Unrecognized parameter in "${SecondaryDeferredBlockType.PLACEHOLDER}" block: "${param.expression}"`); } } return new t.DeferredBlockPlaceholder(html.visitAll(visitor, ast.children), minimumTime, ast.sourceSpan, ast.startSourceSpan, ast.endSourceSpan); } function parseLoadingBlock(ast, visitor) { let afterTime = null; let minimumTime = null; for (const param of ast.parameters) { if (AFTER_PARAMETER_PATTERN.test(param.expression)) { const parsedTime = parseDeferredTime(param.expression.slice(getTriggerParametersStart(param.expression))); if (parsedTime === null) { throw new Error(`Could not parse time value of parameter "after"`); } afterTime = parsedTime; } else if (MINIMUM_PARAMETER_PATTERN.test(param.expression)) { const parsedTime = parseDeferredTime(param.expression.slice(getTriggerParametersStart(param.expression))); if (parsedTime === null) { throw new Error(`Could not parse time value of parameter "minimum"`); } minimumTime = parsedTime; } else { throw new Error(`Unrecognized parameter in "${SecondaryDeferredBlockType.LOADING}" block: "${param.expression}"`); } } return new t.DeferredBlockLoading(html.visitAll(visitor, ast.children), afterTime, minimumTime, ast.sourceSpan, ast.startSourceSpan, ast.endSourceSpan); } function parseErrorBlock(ast, visitor) { if (ast.parameters.length > 0) { throw new Error(`"${SecondaryDeferredBlockType.ERROR}" block cannot have parameters`); } return new t.DeferredBlockError(html.visitAll(visitor, ast.children), ast.sourceSpan, ast.startSourceSpan, ast.endSourceSpan); } function parsePrimaryTriggers(params, bindingParser, errors) { const triggers = []; const prefetchTriggers = []; for (const param of params) { // The lexer ignores the leading spaces so we can assume // that the expression starts with a keyword. if (WHEN_PARAMETER_PATTERN.test(param.expression)) { const result = parseWhenTrigger(param, bindingParser, errors); result !== null && triggers.push(result); } else if (ON_PARAMETER_PATTERN.test(param.expression)) { triggers.push(...parseOnTrigger(param, errors)); } else if (PREFETCH_WHEN_PATTERN.test(param.expression)) { const result = parseWhenTrigger(param, bindingParser, errors); result !== null && prefetchTriggers.push(result); } else if (PREFETCH_ON_PATTERN.test(param.expression)) { prefetchTriggers.push(...parseOnTrigger(param, errors)); } else { errors.push(new ParseError(param.sourceSpan, 'Unrecognized trigger')); } } return { triggers, prefetchTriggers }; } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"r3_deferred_blocks.js","sourceRoot":"","sources":["../../../../../../../packages/compiler/src/render3/r3_deferred_blocks.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 * as html from '../ml_parser/ast';\nimport {ParseError} from '../parse_util';\nimport {BindingParser} from '../template_parser/binding_parser';\n\nimport * as t from './r3_ast';\nimport {getTriggerParametersStart, parseDeferredTime, parseOnTrigger, parseWhenTrigger} from './r3_deferred_triggers';\n\n/** Pattern to identify a `prefetch when` trigger. */\nconst PREFETCH_WHEN_PATTERN = /^prefetch\\s+when\\s/;\n\n/** Pattern to identify a `prefetch on` trigger. */\nconst PREFETCH_ON_PATTERN = /^prefetch\\s+on\\s/;\n\n/** Pattern to identify a `minimum` parameter in a block. */\nconst MINIMUM_PARAMETER_PATTERN = /^minimum\\s/;\n\n/** Pattern to identify a `after` parameter in a block. */\nconst AFTER_PARAMETER_PATTERN = /^after\\s/;\n\n/** Pattern to identify a `when` parameter in a block. */\nconst WHEN_PARAMETER_PATTERN = /^when\\s/;\n\n/** Pattern to identify a `on` parameter in a block. */\nconst ON_PARAMETER_PATTERN = /^on\\s/;\n\n/** Possible types of secondary deferred blocks. */\nexport enum SecondaryDeferredBlockType {\n  PLACEHOLDER = 'placeholder',\n  LOADING = 'loading',\n  ERROR = 'error',\n}\n\n/** Creates a deferred block from an HTML AST node. */\nexport function createDeferredBlock(\n    ast: html.BlockGroup, visitor: html.Visitor,\n    bindingParser: BindingParser): {node: t.DeferredBlock, errors: ParseError[]} {\n  const errors: ParseError[] = [];\n  const [primaryBlock, ...secondaryBlocks] = ast.blocks;\n  const {triggers, prefetchTriggers} =\n      parsePrimaryTriggers(primaryBlock.parameters, bindingParser, errors);\n  const {placeholder, loading, error} = parseSecondaryBlocks(secondaryBlocks, errors, visitor);\n\n  return {\n    node: new t.DeferredBlock(\n        html.visitAll(visitor, primaryBlock.children), triggers, prefetchTriggers, placeholder,\n        loading, error, ast.sourceSpan, ast.startSourceSpan, ast.endSourceSpan),\n    errors,\n  };\n}\n\nfunction parseSecondaryBlocks(blocks: html.Block[], errors: ParseError[], visitor: html.Visitor) {\n  let placeholder: t.DeferredBlockPlaceholder|null = null;\n  let loading: t.DeferredBlockLoading|null = null;\n  let error: t.DeferredBlockError|null = null;\n\n  for (const block of blocks) {\n    try {\n      switch (block.name) {\n        case SecondaryDeferredBlockType.PLACEHOLDER:\n          if (placeholder !== null) {\n            errors.push(new ParseError(\n                block.startSourceSpan,\n                `\"defer\" block can only have one \"${\n                    SecondaryDeferredBlockType.PLACEHOLDER}\" block`));\n          } else {\n            placeholder = parsePlaceholderBlock(block, visitor);\n          }\n          break;\n\n        case SecondaryDeferredBlockType.LOADING:\n          if (loading !== null) {\n            errors.push(new ParseError(\n                block.startSourceSpan,\n                `\"defer\" block can only have one \"${SecondaryDeferredBlockType.LOADING}\" block`));\n          } else {\n            loading = parseLoadingBlock(block, visitor);\n          }\n          break;\n\n        case SecondaryDeferredBlockType.ERROR:\n          if (error !== null) {\n            errors.push(new ParseError(\n                block.startSourceSpan,\n                `\"defer\" block can only have one \"${SecondaryDeferredBlockType.ERROR}\" block`));\n          } else {\n            error = parseErrorBlock(block, visitor);\n          }\n          break;\n\n        default:\n          errors.push(new ParseError(block.startSourceSpan, `Unrecognized block \"${block.name}\"`));\n          break;\n      }\n    } catch (e) {\n      errors.push(new ParseError(block.startSourceSpan, (e as Error).message));\n    }\n  }\n\n  return {placeholder, loading, error};\n}\n\nfunction parsePlaceholderBlock(ast: html.Block, visitor: html.Visitor): t.DeferredBlockPlaceholder {\n  let minimumTime: number|null = null;\n\n  for (const param of ast.parameters) {\n    if (MINIMUM_PARAMETER_PATTERN.test(param.expression)) {\n      const parsedTime =\n          parseDeferredTime(param.expression.slice(getTriggerParametersStart(param.expression)));\n\n      if (parsedTime === null) {\n        throw new Error(`Could not parse time value of parameter \"minimum\"`);\n      }\n\n      minimumTime = parsedTime;\n    } else {\n      throw new Error(`Unrecognized parameter in \"${\n          SecondaryDeferredBlockType.PLACEHOLDER}\" block: \"${param.expression}\"`);\n    }\n  }\n\n  return new t.DeferredBlockPlaceholder(\n      html.visitAll(visitor, ast.children), minimumTime, ast.sourceSpan, ast.startSourceSpan,\n      ast.endSourceSpan);\n}\n\nfunction parseLoadingBlock(ast: html.Block, visitor: html.Visitor): t.DeferredBlockLoading {\n  let afterTime: number|null = null;\n  let minimumTime: number|null = null;\n\n  for (const param of ast.parameters) {\n    if (AFTER_PARAMETER_PATTERN.test(param.expression)) {\n      const parsedTime =\n          parseDeferredTime(param.expression.slice(getTriggerParametersStart(param.expression)));\n\n      if (parsedTime === null) {\n        throw new Error(`Could not parse time value of parameter \"after\"`);\n      }\n\n      afterTime = parsedTime;\n    } else if (MINIMUM_PARAMETER_PATTERN.test(param.expression)) {\n      const parsedTime =\n          parseDeferredTime(param.expression.slice(getTriggerParametersStart(param.expression)));\n\n      if (parsedTime === null) {\n        throw new Error(`Could not parse time value of parameter \"minimum\"`);\n      }\n\n      minimumTime = parsedTime;\n    } else {\n      throw new Error(`Unrecognized parameter in \"${SecondaryDeferredBlockType.LOADING}\" block: \"${\n          param.expression}\"`);\n    }\n  }\n\n  return new t.DeferredBlockLoading(\n      html.visitAll(visitor, ast.children), afterTime, minimumTime, ast.sourceSpan,\n      ast.startSourceSpan, ast.endSourceSpan);\n}\n\n\nfunction parseErrorBlock(ast: html.Block, visitor: html.Visitor): t.DeferredBlockError {\n  if (ast.parameters.length > 0) {\n    throw new Error(`\"${SecondaryDeferredBlockType.ERROR}\" block cannot have parameters`);\n  }\n\n  return new t.DeferredBlockError(\n      html.visitAll(visitor, ast.children), ast.sourceSpan, ast.startSourceSpan, ast.endSourceSpan);\n}\n\nfunction parsePrimaryTriggers(\n    params: html.BlockParameter[], bindingParser: BindingParser, errors: ParseError[]) {\n  const triggers: t.DeferredTrigger[] = [];\n  const prefetchTriggers: t.DeferredTrigger[] = [];\n\n  for (const param of params) {\n    // The lexer ignores the leading spaces so we can assume\n    // that the expression starts with a keyword.\n    if (WHEN_PARAMETER_PATTERN.test(param.expression)) {\n      const result = parseWhenTrigger(param, bindingParser, errors);\n      result !== null && triggers.push(result);\n    } else if (ON_PARAMETER_PATTERN.test(param.expression)) {\n      triggers.push(...parseOnTrigger(param, errors));\n    } else if (PREFETCH_WHEN_PATTERN.test(param.expression)) {\n      const result = parseWhenTrigger(param, bindingParser, errors);\n      result !== null && prefetchTriggers.push(result);\n    } else if (PREFETCH_ON_PATTERN.test(param.expression)) {\n      prefetchTriggers.push(...parseOnTrigger(param, errors));\n    } else {\n      errors.push(new ParseError(param.sourceSpan, 'Unrecognized trigger'));\n    }\n  }\n\n  return {triggers, prefetchTriggers};\n}\n"]}