@angular/compiler 17.3.1 → 18.0.0-next.1
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/esm2022/src/compiler.mjs +2 -2
- package/esm2022/src/expression_parser/ast.mjs +25 -1
- package/esm2022/src/jit_compiler_facade.mjs +13 -32
- package/esm2022/src/render3/partial/class_metadata.mjs +1 -1
- package/esm2022/src/render3/partial/directive.mjs +1 -1
- package/esm2022/src/render3/partial/factory.mjs +1 -1
- package/esm2022/src/render3/partial/injectable.mjs +1 -1
- package/esm2022/src/render3/partial/injector.mjs +1 -1
- package/esm2022/src/render3/partial/ng_module.mjs +1 -1
- package/esm2022/src/render3/partial/pipe.mjs +1 -1
- package/esm2022/src/render3/r3_ast.mjs +4 -3
- package/esm2022/src/render3/r3_class_metadata_compiler.mjs +3 -3
- package/esm2022/src/render3/r3_template_transform.mjs +3 -2
- package/esm2022/src/render3/view/api.mjs +1 -1
- package/esm2022/src/render3/view/compiler.mjs +89 -353
- package/esm2022/src/render3/view/i18n/util.mjs +1 -85
- package/esm2022/src/render3/view/query_generation.mjs +5 -2
- package/esm2022/src/render3/view/template.mjs +1 -2248
- package/esm2022/src/render3/view/util.mjs +3 -134
- package/esm2022/src/template/pipeline/ir/src/ops/create.mjs +3 -3
- package/esm2022/src/template/pipeline/src/compilation.mjs +3 -3
- package/esm2022/src/template/pipeline/src/emit.mjs +3 -3
- package/esm2022/src/template/pipeline/src/ingest.mjs +39 -30
- package/esm2022/src/template/pipeline/src/phases/i18n_const_collection.mjs +20 -2
- package/esm2022/src/template/pipeline/src/phases/naming.mjs +2 -2
- package/esm2022/src/template/pipeline/src/phases/parse_extracted_styles.mjs +81 -3
- package/esm2022/src/template/pipeline/src/phases/resolve_defer_deps_fns.mjs +31 -0
- package/esm2022/src/template_parser/binding_parser.mjs +9 -9
- package/esm2022/src/version.mjs +1 -1
- package/fesm2022/compiler.mjs +11608 -15859
- package/fesm2022/compiler.mjs.map +1 -1
- package/index.d.ts +54 -45
- package/package.json +2 -2
- package/esm2022/src/compiler_util/expression_converter.mjs +0 -852
- package/esm2022/src/render3/view/i18n/context.mjs +0 -212
- package/esm2022/src/render3/view/style_parser.mjs +0 -87
- package/esm2022/src/render3/view/styling_builder.mjs +0 -534
- package/esm2022/src/template/pipeline/src/phases/create_defer_deps_fns.mjs +0 -50
- package/esm2022/src/template/pipeline/switch/index.mjs +0 -2
|
@@ -1,212 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* @license
|
|
3
|
-
* Copyright Google LLC All Rights Reserved.
|
|
4
|
-
*
|
|
5
|
-
* Use of this source code is governed by an MIT-style license that can be
|
|
6
|
-
* found in the LICENSE file at https://angular.io/license
|
|
7
|
-
*/
|
|
8
|
-
import { assembleBoundTextPlaceholders, getSeqNumberGenerator, updatePlaceholderMap, wrapI18nPlaceholder } from './util';
|
|
9
|
-
var TagType;
|
|
10
|
-
(function (TagType) {
|
|
11
|
-
TagType[TagType["ELEMENT"] = 0] = "ELEMENT";
|
|
12
|
-
TagType[TagType["TEMPLATE"] = 1] = "TEMPLATE";
|
|
13
|
-
})(TagType || (TagType = {}));
|
|
14
|
-
/**
|
|
15
|
-
* Generates an object that is used as a shared state between parent and all child contexts.
|
|
16
|
-
*/
|
|
17
|
-
function setupRegistry() {
|
|
18
|
-
return { getUniqueId: getSeqNumberGenerator(), icus: new Map() };
|
|
19
|
-
}
|
|
20
|
-
/**
|
|
21
|
-
* I18nContext is a helper class which keeps track of all i18n-related aspects
|
|
22
|
-
* (accumulates placeholders, bindings, etc) between i18nStart and i18nEnd instructions.
|
|
23
|
-
*
|
|
24
|
-
* When we enter a nested template, the top-level context is being passed down
|
|
25
|
-
* to the nested component, which uses this context to generate a child instance
|
|
26
|
-
* of I18nContext class (to handle nested template) and at the end, reconciles it back
|
|
27
|
-
* with the parent context.
|
|
28
|
-
*
|
|
29
|
-
* @param index Instruction index of i18nStart, which initiates this context
|
|
30
|
-
* @param ref Reference to a translation const that represents the content if thus context
|
|
31
|
-
* @param level Nesting level defined for child contexts
|
|
32
|
-
* @param templateIndex Instruction index of a template which this context belongs to
|
|
33
|
-
* @param meta Meta information (id, meaning, description, etc) associated with this context
|
|
34
|
-
*/
|
|
35
|
-
export class I18nContext {
|
|
36
|
-
constructor(index, ref, level = 0, templateIndex = null, meta, registry) {
|
|
37
|
-
this.index = index;
|
|
38
|
-
this.ref = ref;
|
|
39
|
-
this.level = level;
|
|
40
|
-
this.templateIndex = templateIndex;
|
|
41
|
-
this.meta = meta;
|
|
42
|
-
this.registry = registry;
|
|
43
|
-
this.bindings = new Set();
|
|
44
|
-
this.placeholders = new Map();
|
|
45
|
-
this.isEmitted = false;
|
|
46
|
-
this._unresolvedCtxCount = 0;
|
|
47
|
-
this._registry = registry || setupRegistry();
|
|
48
|
-
this.id = this._registry.getUniqueId();
|
|
49
|
-
}
|
|
50
|
-
appendTag(type, node, index, closed) {
|
|
51
|
-
if (node.isVoid && closed) {
|
|
52
|
-
return; // ignore "close" for void tags
|
|
53
|
-
}
|
|
54
|
-
const ph = node.isVoid || !closed ? node.startName : node.closeName;
|
|
55
|
-
const content = { type, index, ctx: this.id, isVoid: node.isVoid, closed };
|
|
56
|
-
updatePlaceholderMap(this.placeholders, ph, content);
|
|
57
|
-
}
|
|
58
|
-
appendBlockPart(node, index, closed) {
|
|
59
|
-
const ph = closed ? node.closeName : node.startName;
|
|
60
|
-
const content = { type: TagType.TEMPLATE, index, ctx: this.id, closed };
|
|
61
|
-
updatePlaceholderMap(this.placeholders, ph, content);
|
|
62
|
-
}
|
|
63
|
-
get icus() {
|
|
64
|
-
return this._registry.icus;
|
|
65
|
-
}
|
|
66
|
-
get isRoot() {
|
|
67
|
-
return this.level === 0;
|
|
68
|
-
}
|
|
69
|
-
get isResolved() {
|
|
70
|
-
return this._unresolvedCtxCount === 0;
|
|
71
|
-
}
|
|
72
|
-
getSerializedPlaceholders() {
|
|
73
|
-
const result = new Map();
|
|
74
|
-
this.placeholders.forEach((values, key) => result.set(key, values.map(serializePlaceholderValue)));
|
|
75
|
-
return result;
|
|
76
|
-
}
|
|
77
|
-
// public API to accumulate i18n-related content
|
|
78
|
-
appendBinding(binding) {
|
|
79
|
-
this.bindings.add(binding);
|
|
80
|
-
}
|
|
81
|
-
appendIcu(name, ref) {
|
|
82
|
-
updatePlaceholderMap(this._registry.icus, name, ref);
|
|
83
|
-
}
|
|
84
|
-
appendBoundText(node) {
|
|
85
|
-
const phs = assembleBoundTextPlaceholders(node, this.bindings.size, this.id);
|
|
86
|
-
phs.forEach((values, key) => updatePlaceholderMap(this.placeholders, key, ...values));
|
|
87
|
-
}
|
|
88
|
-
appendTemplate(node, index) {
|
|
89
|
-
// add open and close tags at the same time,
|
|
90
|
-
// since we process nested templates separately
|
|
91
|
-
this.appendTag(TagType.TEMPLATE, node, index, false);
|
|
92
|
-
this.appendTag(TagType.TEMPLATE, node, index, true);
|
|
93
|
-
this._unresolvedCtxCount++;
|
|
94
|
-
}
|
|
95
|
-
appendBlock(node, index) {
|
|
96
|
-
// add open and close tags at the same time,
|
|
97
|
-
// since we process nested templates separately
|
|
98
|
-
this.appendBlockPart(node, index, false);
|
|
99
|
-
this.appendBlockPart(node, index, true);
|
|
100
|
-
this._unresolvedCtxCount++;
|
|
101
|
-
}
|
|
102
|
-
appendElement(node, index, closed) {
|
|
103
|
-
this.appendTag(TagType.ELEMENT, node, index, closed);
|
|
104
|
-
}
|
|
105
|
-
appendProjection(node, index) {
|
|
106
|
-
// Add open and close tags at the same time, since `<ng-content>` has no content,
|
|
107
|
-
// so when we come across `<ng-content>` we can register both open and close tags.
|
|
108
|
-
// Note: runtime i18n logic doesn't distinguish `<ng-content>` tag placeholders and
|
|
109
|
-
// regular element tag placeholders, so we generate element placeholders for both types.
|
|
110
|
-
this.appendTag(TagType.ELEMENT, node, index, false);
|
|
111
|
-
this.appendTag(TagType.ELEMENT, node, index, true);
|
|
112
|
-
}
|
|
113
|
-
/**
|
|
114
|
-
* Generates an instance of a child context based on the root one,
|
|
115
|
-
* when we enter a nested template within I18n section.
|
|
116
|
-
*
|
|
117
|
-
* @param index Instruction index of corresponding i18nStart, which initiates this context
|
|
118
|
-
* @param templateIndex Instruction index of a template which this context belongs to
|
|
119
|
-
* @param meta Meta information (id, meaning, description, etc) associated with this context
|
|
120
|
-
*
|
|
121
|
-
* @returns I18nContext instance
|
|
122
|
-
*/
|
|
123
|
-
forkChildContext(index, templateIndex, meta) {
|
|
124
|
-
return new I18nContext(index, this.ref, this.level + 1, templateIndex, meta, this._registry);
|
|
125
|
-
}
|
|
126
|
-
/**
|
|
127
|
-
* Reconciles child context into parent one once the end of the i18n block is reached (i18nEnd).
|
|
128
|
-
*
|
|
129
|
-
* @param context Child I18nContext instance to be reconciled with parent context.
|
|
130
|
-
*/
|
|
131
|
-
reconcileChildContext(context) {
|
|
132
|
-
// set the right context id for open and close
|
|
133
|
-
// template tags, so we can use it as sub-block ids
|
|
134
|
-
['start', 'close'].forEach((op) => {
|
|
135
|
-
const key = context.meta[`${op}Name`];
|
|
136
|
-
const phs = this.placeholders.get(key) || [];
|
|
137
|
-
const tag = phs.find(findTemplateFn(this.id, context.templateIndex));
|
|
138
|
-
if (tag) {
|
|
139
|
-
tag.ctx = context.id;
|
|
140
|
-
}
|
|
141
|
-
});
|
|
142
|
-
// reconcile placeholders
|
|
143
|
-
const childPhs = context.placeholders;
|
|
144
|
-
childPhs.forEach((values, key) => {
|
|
145
|
-
const phs = this.placeholders.get(key);
|
|
146
|
-
if (!phs) {
|
|
147
|
-
this.placeholders.set(key, values);
|
|
148
|
-
return;
|
|
149
|
-
}
|
|
150
|
-
// try to find matching template...
|
|
151
|
-
const tmplIdx = phs.findIndex(findTemplateFn(context.id, context.templateIndex));
|
|
152
|
-
if (tmplIdx >= 0) {
|
|
153
|
-
// ... if found - replace it with nested template content
|
|
154
|
-
const isCloseTag = key.startsWith('CLOSE');
|
|
155
|
-
const isTemplateTag = key.endsWith('NG-TEMPLATE');
|
|
156
|
-
if (isTemplateTag) {
|
|
157
|
-
// current template's content is placed before or after
|
|
158
|
-
// parent template tag, depending on the open/close attribute
|
|
159
|
-
phs.splice(tmplIdx + (isCloseTag ? 0 : 1), 0, ...values);
|
|
160
|
-
}
|
|
161
|
-
else {
|
|
162
|
-
const idx = isCloseTag ? values.length - 1 : 0;
|
|
163
|
-
values[idx].tmpl = phs[tmplIdx];
|
|
164
|
-
phs.splice(tmplIdx, 1, ...values);
|
|
165
|
-
}
|
|
166
|
-
}
|
|
167
|
-
else {
|
|
168
|
-
// ... otherwise just append content to placeholder value
|
|
169
|
-
phs.push(...values);
|
|
170
|
-
}
|
|
171
|
-
this.placeholders.set(key, phs);
|
|
172
|
-
});
|
|
173
|
-
this._unresolvedCtxCount--;
|
|
174
|
-
}
|
|
175
|
-
}
|
|
176
|
-
//
|
|
177
|
-
// Helper methods
|
|
178
|
-
//
|
|
179
|
-
function wrap(symbol, index, contextId, closed) {
|
|
180
|
-
const state = closed ? '/' : '';
|
|
181
|
-
return wrapI18nPlaceholder(`${state}${symbol}${index}`, contextId);
|
|
182
|
-
}
|
|
183
|
-
function wrapTag(symbol, { index, ctx, isVoid }, closed) {
|
|
184
|
-
return isVoid ? wrap(symbol, index, ctx) + wrap(symbol, index, ctx, true) :
|
|
185
|
-
wrap(symbol, index, ctx, closed);
|
|
186
|
-
}
|
|
187
|
-
function findTemplateFn(ctx, templateIndex) {
|
|
188
|
-
return (token) => typeof token === 'object' && token.type === TagType.TEMPLATE &&
|
|
189
|
-
token.index === templateIndex && token.ctx === ctx;
|
|
190
|
-
}
|
|
191
|
-
function serializePlaceholderValue(value) {
|
|
192
|
-
const element = (data, closed) => wrapTag('#', data, closed);
|
|
193
|
-
const template = (data, closed) => wrapTag('*', data, closed);
|
|
194
|
-
switch (value.type) {
|
|
195
|
-
case TagType.ELEMENT:
|
|
196
|
-
// close element tag
|
|
197
|
-
if (value.closed) {
|
|
198
|
-
return element(value, true) + (value.tmpl ? template(value.tmpl, true) : '');
|
|
199
|
-
}
|
|
200
|
-
// open element tag that also initiates a template
|
|
201
|
-
if (value.tmpl) {
|
|
202
|
-
return template(value.tmpl) + element(value) +
|
|
203
|
-
(value.isVoid ? template(value.tmpl, true) : '');
|
|
204
|
-
}
|
|
205
|
-
return element(value);
|
|
206
|
-
case TagType.TEMPLATE:
|
|
207
|
-
return template(value, value.closed);
|
|
208
|
-
default:
|
|
209
|
-
return value;
|
|
210
|
-
}
|
|
211
|
-
}
|
|
212
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"context.js","sourceRoot":"","sources":["../../../../../../../../../packages/compiler/src/render3/view/i18n/context.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 {AST} from '../../../expression_parser/ast';\nimport * as i18n from '../../../i18n/i18n_ast';\nimport * as o from '../../../output/output_ast';\n\nimport {assembleBoundTextPlaceholders, getSeqNumberGenerator, updatePlaceholderMap, wrapI18nPlaceholder} from './util';\n\nenum TagType {\n  ELEMENT,\n  TEMPLATE,\n}\n\n/**\n * Generates an object that is used as a shared state between parent and all child contexts.\n */\nfunction setupRegistry() {\n  return {getUniqueId: getSeqNumberGenerator(), icus: new Map<string, any[]>()};\n}\n\n/**\n * I18nContext is a helper class which keeps track of all i18n-related aspects\n * (accumulates placeholders, bindings, etc) between i18nStart and i18nEnd instructions.\n *\n * When we enter a nested template, the top-level context is being passed down\n * to the nested component, which uses this context to generate a child instance\n * of I18nContext class (to handle nested template) and at the end, reconciles it back\n * with the parent context.\n *\n * @param index Instruction index of i18nStart, which initiates this context\n * @param ref Reference to a translation const that represents the content if thus context\n * @param level Nesting level defined for child contexts\n * @param templateIndex Instruction index of a template which this context belongs to\n * @param meta Meta information (id, meaning, description, etc) associated with this context\n */\nexport class I18nContext {\n  public readonly id: number;\n  public bindings = new Set<AST>();\n  public placeholders = new Map<string, any[]>();\n  public isEmitted: boolean = false;\n\n  private _registry!: any;\n  private _unresolvedCtxCount: number = 0;\n\n  constructor(\n      readonly index: number, readonly ref: o.ReadVarExpr, readonly level: number = 0,\n      readonly templateIndex: number|null = null, readonly meta: i18n.I18nMeta,\n      private registry?: any) {\n    this._registry = registry || setupRegistry();\n    this.id = this._registry.getUniqueId();\n  }\n\n  private appendTag(type: TagType, node: i18n.TagPlaceholder, index: number, closed?: boolean) {\n    if (node.isVoid && closed) {\n      return;  // ignore \"close\" for void tags\n    }\n    const ph = node.isVoid || !closed ? node.startName : node.closeName;\n    const content = {type, index, ctx: this.id, isVoid: node.isVoid, closed};\n    updatePlaceholderMap(this.placeholders, ph, content);\n  }\n\n  private appendBlockPart(node: i18n.BlockPlaceholder, index: number, closed?: boolean) {\n    const ph = closed ? node.closeName : node.startName;\n    const content = {type: TagType.TEMPLATE, index, ctx: this.id, closed};\n    updatePlaceholderMap(this.placeholders, ph, content);\n  }\n\n  get icus() {\n    return this._registry.icus;\n  }\n  get isRoot() {\n    return this.level === 0;\n  }\n  get isResolved() {\n    return this._unresolvedCtxCount === 0;\n  }\n\n  getSerializedPlaceholders() {\n    const result = new Map<string, any[]>();\n    this.placeholders.forEach(\n        (values, key) => result.set(key, values.map(serializePlaceholderValue)));\n    return result;\n  }\n\n  // public API to accumulate i18n-related content\n  appendBinding(binding: AST) {\n    this.bindings.add(binding);\n  }\n  appendIcu(name: string, ref: o.Expression) {\n    updatePlaceholderMap(this._registry.icus, name, ref);\n  }\n  appendBoundText(node: i18n.I18nMeta) {\n    const phs = assembleBoundTextPlaceholders(node, this.bindings.size, this.id);\n    phs.forEach((values, key) => updatePlaceholderMap(this.placeholders, key, ...values));\n  }\n  appendTemplate(node: i18n.I18nMeta, index: number) {\n    // add open and close tags at the same time,\n    // since we process nested templates separately\n    this.appendTag(TagType.TEMPLATE, node as i18n.TagPlaceholder, index, false);\n    this.appendTag(TagType.TEMPLATE, node as i18n.TagPlaceholder, index, true);\n    this._unresolvedCtxCount++;\n  }\n  appendBlock(node: i18n.BlockPlaceholder, index: number) {\n    // add open and close tags at the same time,\n    // since we process nested templates separately\n    this.appendBlockPart(node, index, false);\n    this.appendBlockPart(node, index, true);\n    this._unresolvedCtxCount++;\n  }\n  appendElement(node: i18n.I18nMeta, index: number, closed?: boolean) {\n    this.appendTag(TagType.ELEMENT, node as i18n.TagPlaceholder, index, closed);\n  }\n  appendProjection(node: i18n.I18nMeta, index: number) {\n    // Add open and close tags at the same time, since `<ng-content>` has no content,\n    // so when we come across `<ng-content>` we can register both open and close tags.\n    // Note: runtime i18n logic doesn't distinguish `<ng-content>` tag placeholders and\n    // regular element tag placeholders, so we generate element placeholders for both types.\n    this.appendTag(TagType.ELEMENT, node as i18n.TagPlaceholder, index, false);\n    this.appendTag(TagType.ELEMENT, node as i18n.TagPlaceholder, index, true);\n  }\n\n  /**\n   * Generates an instance of a child context based on the root one,\n   * when we enter a nested template within I18n section.\n   *\n   * @param index Instruction index of corresponding i18nStart, which initiates this context\n   * @param templateIndex Instruction index of a template which this context belongs to\n   * @param meta Meta information (id, meaning, description, etc) associated with this context\n   *\n   * @returns I18nContext instance\n   */\n  forkChildContext(index: number, templateIndex: number, meta: i18n.I18nMeta) {\n    return new I18nContext(index, this.ref, this.level + 1, templateIndex, meta, this._registry);\n  }\n\n  /**\n   * Reconciles child context into parent one once the end of the i18n block is reached (i18nEnd).\n   *\n   * @param context Child I18nContext instance to be reconciled with parent context.\n   */\n  reconcileChildContext(context: I18nContext) {\n    // set the right context id for open and close\n    // template tags, so we can use it as sub-block ids\n    ['start', 'close'].forEach((op: string) => {\n      const key = (context.meta as any)[`${op}Name`];\n      const phs = this.placeholders.get(key) || [];\n      const tag = phs.find(findTemplateFn(this.id, context.templateIndex));\n      if (tag) {\n        tag.ctx = context.id;\n      }\n    });\n\n    // reconcile placeholders\n    const childPhs = context.placeholders;\n    childPhs.forEach((values: any[], key: string) => {\n      const phs = this.placeholders.get(key);\n      if (!phs) {\n        this.placeholders.set(key, values);\n        return;\n      }\n      // try to find matching template...\n      const tmplIdx = phs.findIndex(findTemplateFn(context.id, context.templateIndex));\n      if (tmplIdx >= 0) {\n        // ... if found - replace it with nested template content\n        const isCloseTag = key.startsWith('CLOSE');\n        const isTemplateTag = key.endsWith('NG-TEMPLATE');\n        if (isTemplateTag) {\n          // current template's content is placed before or after\n          // parent template tag, depending on the open/close attribute\n          phs.splice(tmplIdx + (isCloseTag ? 0 : 1), 0, ...values);\n        } else {\n          const idx = isCloseTag ? values.length - 1 : 0;\n          values[idx].tmpl = phs[tmplIdx];\n          phs.splice(tmplIdx, 1, ...values);\n        }\n      } else {\n        // ... otherwise just append content to placeholder value\n        phs.push(...values);\n      }\n      this.placeholders.set(key, phs);\n    });\n    this._unresolvedCtxCount--;\n  }\n}\n\n//\n// Helper methods\n//\n\nfunction wrap(symbol: string, index: number, contextId: number, closed?: boolean): string {\n  const state = closed ? '/' : '';\n  return wrapI18nPlaceholder(`${state}${symbol}${index}`, contextId);\n}\n\nfunction wrapTag(symbol: string, {index, ctx, isVoid}: any, closed?: boolean): string {\n  return isVoid ? wrap(symbol, index, ctx) + wrap(symbol, index, ctx, true) :\n                  wrap(symbol, index, ctx, closed);\n}\n\nfunction findTemplateFn(ctx: number, templateIndex: number|null) {\n  return (token: any) => typeof token === 'object' && token.type === TagType.TEMPLATE &&\n      token.index === templateIndex && token.ctx === ctx;\n}\n\nfunction serializePlaceholderValue(value: any): string {\n  const element = (data: any, closed?: boolean) => wrapTag('#', data, closed);\n  const template = (data: any, closed?: boolean) => wrapTag('*', data, closed);\n\n  switch (value.type) {\n    case TagType.ELEMENT:\n      // close element tag\n      if (value.closed) {\n        return element(value, true) + (value.tmpl ? template(value.tmpl, true) : '');\n      }\n      // open element tag that also initiates a template\n      if (value.tmpl) {\n        return template(value.tmpl) + element(value) +\n            (value.isVoid ? template(value.tmpl, true) : '');\n      }\n      return element(value);\n\n    case TagType.TEMPLATE:\n      return template(value, value.closed);\n\n    default:\n      return value;\n  }\n}\n"]}
|
|
@@ -1,87 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* @license
|
|
3
|
-
* Copyright Google LLC All Rights Reserved.
|
|
4
|
-
*
|
|
5
|
-
* Use of this source code is governed by an MIT-style license that can be
|
|
6
|
-
* found in the LICENSE file at https://angular.io/license
|
|
7
|
-
*/
|
|
8
|
-
/**
|
|
9
|
-
* Parses string representation of a style and converts it into object literal.
|
|
10
|
-
*
|
|
11
|
-
* @param value string representation of style as used in the `style` attribute in HTML.
|
|
12
|
-
* Example: `color: red; height: auto`.
|
|
13
|
-
* @returns An array of style property name and value pairs, e.g. `['color', 'red', 'height',
|
|
14
|
-
* 'auto']`
|
|
15
|
-
*/
|
|
16
|
-
export function parse(value) {
|
|
17
|
-
// we use a string array here instead of a string map
|
|
18
|
-
// because a string-map is not guaranteed to retain the
|
|
19
|
-
// order of the entries whereas a string array can be
|
|
20
|
-
// constructed in a [key, value, key, value] format.
|
|
21
|
-
const styles = [];
|
|
22
|
-
let i = 0;
|
|
23
|
-
let parenDepth = 0;
|
|
24
|
-
let quote = 0 /* Char.QuoteNone */;
|
|
25
|
-
let valueStart = 0;
|
|
26
|
-
let propStart = 0;
|
|
27
|
-
let currentProp = null;
|
|
28
|
-
while (i < value.length) {
|
|
29
|
-
const token = value.charCodeAt(i++);
|
|
30
|
-
switch (token) {
|
|
31
|
-
case 40 /* Char.OpenParen */:
|
|
32
|
-
parenDepth++;
|
|
33
|
-
break;
|
|
34
|
-
case 41 /* Char.CloseParen */:
|
|
35
|
-
parenDepth--;
|
|
36
|
-
break;
|
|
37
|
-
case 39 /* Char.QuoteSingle */:
|
|
38
|
-
// valueStart needs to be there since prop values don't
|
|
39
|
-
// have quotes in CSS
|
|
40
|
-
if (quote === 0 /* Char.QuoteNone */) {
|
|
41
|
-
quote = 39 /* Char.QuoteSingle */;
|
|
42
|
-
}
|
|
43
|
-
else if (quote === 39 /* Char.QuoteSingle */ && value.charCodeAt(i - 1) !== 92 /* Char.BackSlash */) {
|
|
44
|
-
quote = 0 /* Char.QuoteNone */;
|
|
45
|
-
}
|
|
46
|
-
break;
|
|
47
|
-
case 34 /* Char.QuoteDouble */:
|
|
48
|
-
// same logic as above
|
|
49
|
-
if (quote === 0 /* Char.QuoteNone */) {
|
|
50
|
-
quote = 34 /* Char.QuoteDouble */;
|
|
51
|
-
}
|
|
52
|
-
else if (quote === 34 /* Char.QuoteDouble */ && value.charCodeAt(i - 1) !== 92 /* Char.BackSlash */) {
|
|
53
|
-
quote = 0 /* Char.QuoteNone */;
|
|
54
|
-
}
|
|
55
|
-
break;
|
|
56
|
-
case 58 /* Char.Colon */:
|
|
57
|
-
if (!currentProp && parenDepth === 0 && quote === 0 /* Char.QuoteNone */) {
|
|
58
|
-
// TODO: Do not hyphenate CSS custom property names like: `--intentionallyCamelCase`
|
|
59
|
-
currentProp = hyphenate(value.substring(propStart, i - 1).trim());
|
|
60
|
-
valueStart = i;
|
|
61
|
-
}
|
|
62
|
-
break;
|
|
63
|
-
case 59 /* Char.Semicolon */:
|
|
64
|
-
if (currentProp && valueStart > 0 && parenDepth === 0 && quote === 0 /* Char.QuoteNone */) {
|
|
65
|
-
const styleVal = value.substring(valueStart, i - 1).trim();
|
|
66
|
-
styles.push(currentProp, styleVal);
|
|
67
|
-
propStart = i;
|
|
68
|
-
valueStart = 0;
|
|
69
|
-
currentProp = null;
|
|
70
|
-
}
|
|
71
|
-
break;
|
|
72
|
-
}
|
|
73
|
-
}
|
|
74
|
-
if (currentProp && valueStart) {
|
|
75
|
-
const styleVal = value.slice(valueStart).trim();
|
|
76
|
-
styles.push(currentProp, styleVal);
|
|
77
|
-
}
|
|
78
|
-
return styles;
|
|
79
|
-
}
|
|
80
|
-
export function hyphenate(value) {
|
|
81
|
-
return value
|
|
82
|
-
.replace(/[a-z][A-Z]/g, v => {
|
|
83
|
-
return v.charAt(0) + '-' + v.charAt(1);
|
|
84
|
-
})
|
|
85
|
-
.toLowerCase();
|
|
86
|
-
}
|
|
87
|
-
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3R5bGVfcGFyc2VyLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vLi4vLi4vcGFja2FnZXMvY29tcGlsZXIvc3JjL3JlbmRlcjMvdmlldy9zdHlsZV9wYXJzZXIudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUE7Ozs7OztHQU1HO0FBaUJIOzs7Ozs7O0dBT0c7QUFDSCxNQUFNLFVBQVUsS0FBSyxDQUFDLEtBQWE7SUFDakMscURBQXFEO0lBQ3JELHVEQUF1RDtJQUN2RCxxREFBcUQ7SUFDckQsb0RBQW9EO0lBQ3BELE1BQU0sTUFBTSxHQUFhLEVBQUUsQ0FBQztJQUU1QixJQUFJLENBQUMsR0FBRyxDQUFDLENBQUM7SUFDVixJQUFJLFVBQVUsR0FBRyxDQUFDLENBQUM7SUFDbkIsSUFBSSxLQUFLLHlCQUF1QixDQUFDO0lBQ2pDLElBQUksVUFBVSxHQUFHLENBQUMsQ0FBQztJQUNuQixJQUFJLFNBQVMsR0FBRyxDQUFDLENBQUM7SUFDbEIsSUFBSSxXQUFXLEdBQWdCLElBQUksQ0FBQztJQUNwQyxPQUFPLENBQUMsR0FBRyxLQUFLLENBQUMsTUFBTSxFQUFFLENBQUM7UUFDeEIsTUFBTSxLQUFLLEdBQUcsS0FBSyxDQUFDLFVBQVUsQ0FBQyxDQUFDLEVBQUUsQ0FBUyxDQUFDO1FBQzVDLFFBQVEsS0FBSyxFQUFFLENBQUM7WUFDZDtnQkFDRSxVQUFVLEVBQUUsQ0FBQztnQkFDYixNQUFNO1lBQ1I7Z0JBQ0UsVUFBVSxFQUFFLENBQUM7Z0JBQ2IsTUFBTTtZQUNSO2dCQUNFLHVEQUF1RDtnQkFDdkQscUJBQXFCO2dCQUNyQixJQUFJLEtBQUssMkJBQW1CLEVBQUUsQ0FBQztvQkFDN0IsS0FBSyw0QkFBbUIsQ0FBQztnQkFDM0IsQ0FBQztxQkFBTSxJQUFJLEtBQUssOEJBQXFCLElBQUksS0FBSyxDQUFDLFVBQVUsQ0FBQyxDQUFDLEdBQUcsQ0FBQyxDQUFDLDRCQUFtQixFQUFFLENBQUM7b0JBQ3BGLEtBQUsseUJBQWlCLENBQUM7Z0JBQ3pCLENBQUM7Z0JBQ0QsTUFBTTtZQUNSO2dCQUNFLHNCQUFzQjtnQkFDdEIsSUFBSSxLQUFLLDJCQUFtQixFQUFFLENBQUM7b0JBQzdCLEtBQUssNEJBQW1CLENBQUM7Z0JBQzNCLENBQUM7cUJBQU0sSUFBSSxLQUFLLDhCQUFxQixJQUFJLEtBQUssQ0FBQyxVQUFVLENBQUMsQ0FBQyxHQUFHLENBQUMsQ0FBQyw0QkFBbUIsRUFBRSxDQUFDO29CQUNwRixLQUFLLHlCQUFpQixDQUFDO2dCQUN6QixDQUFDO2dCQUNELE1BQU07WUFDUjtnQkFDRSxJQUFJLENBQUMsV0FBVyxJQUFJLFVBQVUsS0FBSyxDQUFDLElBQUksS0FBSywyQkFBbUIsRUFBRSxDQUFDO29CQUNqRSxvRkFBb0Y7b0JBQ3BGLFdBQVcsR0FBRyxTQUFTLENBQUMsS0FBSyxDQUFDLFNBQVMsQ0FBQyxTQUFTLEVBQUUsQ0FBQyxHQUFHLENBQUMsQ0FBQyxDQUFDLElBQUksRUFBRSxDQUFDLENBQUM7b0JBQ2xFLFVBQVUsR0FBRyxDQUFDLENBQUM7Z0JBQ2pCLENBQUM7Z0JBQ0QsTUFBTTtZQUNSO2dCQUNFLElBQUksV0FBVyxJQUFJLFVBQVUsR0FBRyxDQUFDLElBQUksVUFBVSxLQUFLLENBQUMsSUFBSSxLQUFLLDJCQUFtQixFQUFFLENBQUM7b0JBQ2xGLE1BQU0sUUFBUSxHQUFHLEtBQUssQ0FBQyxTQUFTLENBQUMsVUFBVSxFQUFFLENBQUMsR0FBRyxDQUFDLENBQUMsQ0FBQyxJQUFJLEVBQUUsQ0FBQztvQkFDM0QsTUFBTSxDQUFDLElBQUksQ0FBQyxXQUFXLEVBQUUsUUFBUSxDQUFDLENBQUM7b0JBQ25DLFNBQVMsR0FBRyxDQUFDLENBQUM7b0JBQ2QsVUFBVSxHQUFHLENBQUMsQ0FBQztvQkFDZixXQUFXLEdBQUcsSUFBSSxDQUFDO2dCQUNyQixDQUFDO2dCQUNELE1BQU07UUFDVixDQUFDO0lBQ0gsQ0FBQztJQUVELElBQUksV0FBVyxJQUFJLFVBQVUsRUFBRSxDQUFDO1FBQzlCLE1BQU0sUUFBUSxHQUFHLEtBQUssQ0FBQyxLQUFLLENBQUMsVUFBVSxDQUFDLENBQUMsSUFBSSxFQUFFLENBQUM7UUFDaEQsTUFBTSxDQUFDLElBQUksQ0FBQyxXQUFXLEVBQUUsUUFBUSxDQUFDLENBQUM7SUFDckMsQ0FBQztJQUVELE9BQU8sTUFBTSxDQUFDO0FBQ2hCLENBQUM7QUFFRCxNQUFNLFVBQVUsU0FBUyxDQUFDLEtBQWE7SUFDckMsT0FBTyxLQUFLO1NBQ1AsT0FBTyxDQUNKLGFBQWEsRUFDYixDQUFDLENBQUMsRUFBRTtRQUNGLE9BQU8sQ0FBQyxDQUFDLE1BQU0sQ0FBQyxDQUFDLENBQUMsR0FBRyxHQUFHLEdBQUcsQ0FBQyxDQUFDLE1BQU0sQ0FBQyxDQUFDLENBQUMsQ0FBQztJQUN6QyxDQUFDLENBQUM7U0FDTCxXQUFXLEVBQUUsQ0FBQztBQUNyQixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBAbGljZW5zZVxuICogQ29weXJpZ2h0IEdvb2dsZSBMTEMgQWxsIFJpZ2h0cyBSZXNlcnZlZC5cbiAqXG4gKiBVc2Ugb2YgdGhpcyBzb3VyY2UgY29kZSBpcyBnb3Zlcm5lZCBieSBhbiBNSVQtc3R5bGUgbGljZW5zZSB0aGF0IGNhbiBiZVxuICogZm91bmQgaW4gdGhlIExJQ0VOU0UgZmlsZSBhdCBodHRwczovL2FuZ3VsYXIuaW8vbGljZW5zZVxuICovXG5cbi8vIEFueSBjaGFuZ2VzIGhlcmUgc2hvdWxkIGJlIHBvcnRlZCB0byB0aGUgQW5ndWxhciBEb21pbm8gZm9yay5cbi8vIGh0dHBzOi8vZ2l0aHViLmNvbS9hbmd1bGFyL2RvbWluby9ibG9iL21haW4vbGliL3N0eWxlX3BhcnNlci5qc1xuXG5jb25zdCBlbnVtIENoYXIge1xuICBPcGVuUGFyZW4gPSA0MCxcbiAgQ2xvc2VQYXJlbiA9IDQxLFxuICBDb2xvbiA9IDU4LFxuICBTZW1pY29sb24gPSA1OSxcbiAgQmFja1NsYXNoID0gOTIsXG4gIFF1b3RlTm9uZSA9IDAsICAvLyBpbmRpY2F0aW5nIHdlIGFyZSBub3QgaW5zaWRlIGEgcXVvdGVcbiAgUXVvdGVEb3VibGUgPSAzNCxcbiAgUXVvdGVTaW5nbGUgPSAzOSxcbn1cblxuXG4vKipcbiAqIFBhcnNlcyBzdHJpbmcgcmVwcmVzZW50YXRpb24gb2YgYSBzdHlsZSBhbmQgY29udmVydHMgaXQgaW50byBvYmplY3QgbGl0ZXJhbC5cbiAqXG4gKiBAcGFyYW0gdmFsdWUgc3RyaW5nIHJlcHJlc2VudGF0aW9uIG9mIHN0eWxlIGFzIHVzZWQgaW4gdGhlIGBzdHlsZWAgYXR0cmlidXRlIGluIEhUTUwuXG4gKiAgIEV4YW1wbGU6IGBjb2xvcjogcmVkOyBoZWlnaHQ6IGF1dG9gLlxuICogQHJldHVybnMgQW4gYXJyYXkgb2Ygc3R5bGUgcHJvcGVydHkgbmFtZSBhbmQgdmFsdWUgcGFpcnMsIGUuZy4gYFsnY29sb3InLCAncmVkJywgJ2hlaWdodCcsXG4gKiAnYXV0byddYFxuICovXG5leHBvcnQgZnVuY3Rpb24gcGFyc2UodmFsdWU6IHN0cmluZyk6IHN0cmluZ1tdIHtcbiAgLy8gd2UgdXNlIGEgc3RyaW5nIGFycmF5IGhlcmUgaW5zdGVhZCBvZiBhIHN0cmluZyBtYXBcbiAgLy8gYmVjYXVzZSBhIHN0cmluZy1tYXAgaXMgbm90IGd1YXJhbnRlZWQgdG8gcmV0YWluIHRoZVxuICAvLyBvcmRlciBvZiB0aGUgZW50cmllcyB3aGVyZWFzIGEgc3RyaW5nIGFycmF5IGNhbiBiZVxuICAvLyBjb25zdHJ1Y3RlZCBpbiBhIFtrZXksIHZhbHVlLCBrZXksIHZhbHVlXSBmb3JtYXQuXG4gIGNvbnN0IHN0eWxlczogc3RyaW5nW10gPSBbXTtcblxuICBsZXQgaSA9IDA7XG4gIGxldCBwYXJlbkRlcHRoID0gMDtcbiAgbGV0IHF1b3RlOiBDaGFyID0gQ2hhci5RdW90ZU5vbmU7XG4gIGxldCB2YWx1ZVN0YXJ0ID0gMDtcbiAgbGV0IHByb3BTdGFydCA9IDA7XG4gIGxldCBjdXJyZW50UHJvcDogc3RyaW5nfG51bGwgPSBudWxsO1xuICB3aGlsZSAoaSA8IHZhbHVlLmxlbmd0aCkge1xuICAgIGNvbnN0IHRva2VuID0gdmFsdWUuY2hhckNvZGVBdChpKyspIGFzIENoYXI7XG4gICAgc3dpdGNoICh0b2tlbikge1xuICAgICAgY2FzZSBDaGFyLk9wZW5QYXJlbjpcbiAgICAgICAgcGFyZW5EZXB0aCsrO1xuICAgICAgICBicmVhaztcbiAgICAgIGNhc2UgQ2hhci5DbG9zZVBhcmVuOlxuICAgICAgICBwYXJlbkRlcHRoLS07XG4gICAgICAgIGJyZWFrO1xuICAgICAgY2FzZSBDaGFyLlF1b3RlU2luZ2xlOlxuICAgICAgICAvLyB2YWx1ZVN0YXJ0IG5lZWRzIHRvIGJlIHRoZXJlIHNpbmNlIHByb3AgdmFsdWVzIGRvbid0XG4gICAgICAgIC8vIGhhdmUgcXVvdGVzIGluIENTU1xuICAgICAgICBpZiAocXVvdGUgPT09IENoYXIuUXVvdGVOb25lKSB7XG4gICAgICAgICAgcXVvdGUgPSBDaGFyLlF1b3RlU2luZ2xlO1xuICAgICAgICB9IGVsc2UgaWYgKHF1b3RlID09PSBDaGFyLlF1b3RlU2luZ2xlICYmIHZhbHVlLmNoYXJDb2RlQXQoaSAtIDEpICE9PSBDaGFyLkJhY2tTbGFzaCkge1xuICAgICAgICAgIHF1b3RlID0gQ2hhci5RdW90ZU5vbmU7XG4gICAgICAgIH1cbiAgICAgICAgYnJlYWs7XG4gICAgICBjYXNlIENoYXIuUXVvdGVEb3VibGU6XG4gICAgICAgIC8vIHNhbWUgbG9naWMgYXMgYWJvdmVcbiAgICAgICAgaWYgKHF1b3RlID09PSBDaGFyLlF1b3RlTm9uZSkge1xuICAgICAgICAgIHF1b3RlID0gQ2hhci5RdW90ZURvdWJsZTtcbiAgICAgICAgfSBlbHNlIGlmIChxdW90ZSA9PT0gQ2hhci5RdW90ZURvdWJsZSAmJiB2YWx1ZS5jaGFyQ29kZUF0KGkgLSAxKSAhPT0gQ2hhci5CYWNrU2xhc2gpIHtcbiAgICAgICAgICBxdW90ZSA9IENoYXIuUXVvdGVOb25lO1xuICAgICAgICB9XG4gICAgICAgIGJyZWFrO1xuICAgICAgY2FzZSBDaGFyLkNvbG9uOlxuICAgICAgICBpZiAoIWN1cnJlbnRQcm9wICYmIHBhcmVuRGVwdGggPT09IDAgJiYgcXVvdGUgPT09IENoYXIuUXVvdGVOb25lKSB7XG4gICAgICAgICAgLy8gVE9ETzogRG8gbm90IGh5cGhlbmF0ZSBDU1MgY3VzdG9tIHByb3BlcnR5IG5hbWVzIGxpa2U6IGAtLWludGVudGlvbmFsbHlDYW1lbENhc2VgXG4gICAgICAgICAgY3VycmVudFByb3AgPSBoeXBoZW5hdGUodmFsdWUuc3Vic3RyaW5nKHByb3BTdGFydCwgaSAtIDEpLnRyaW0oKSk7XG4gICAgICAgICAgdmFsdWVTdGFydCA9IGk7XG4gICAgICAgIH1cbiAgICAgICAgYnJlYWs7XG4gICAgICBjYXNlIENoYXIuU2VtaWNvbG9uOlxuICAgICAgICBpZiAoY3VycmVudFByb3AgJiYgdmFsdWVTdGFydCA+IDAgJiYgcGFyZW5EZXB0aCA9PT0gMCAmJiBxdW90ZSA9PT0gQ2hhci5RdW90ZU5vbmUpIHtcbiAgICAgICAgICBjb25zdCBzdHlsZVZhbCA9IHZhbHVlLnN1YnN0cmluZyh2YWx1ZVN0YXJ0LCBpIC0gMSkudHJpbSgpO1xuICAgICAgICAgIHN0eWxlcy5wdXNoKGN1cnJlbnRQcm9wLCBzdHlsZVZhbCk7XG4gICAgICAgICAgcHJvcFN0YXJ0ID0gaTtcbiAgICAgICAgICB2YWx1ZVN0YXJ0ID0gMDtcbiAgICAgICAgICBjdXJyZW50UHJvcCA9IG51bGw7XG4gICAgICAgIH1cbiAgICAgICAgYnJlYWs7XG4gICAgfVxuICB9XG5cbiAgaWYgKGN1cnJlbnRQcm9wICYmIHZhbHVlU3RhcnQpIHtcbiAgICBjb25zdCBzdHlsZVZhbCA9IHZhbHVlLnNsaWNlKHZhbHVlU3RhcnQpLnRyaW0oKTtcbiAgICBzdHlsZXMucHVzaChjdXJyZW50UHJvcCwgc3R5bGVWYWwpO1xuICB9XG5cbiAgcmV0dXJuIHN0eWxlcztcbn1cblxuZXhwb3J0IGZ1bmN0aW9uIGh5cGhlbmF0ZSh2YWx1ZTogc3RyaW5nKTogc3RyaW5nIHtcbiAgcmV0dXJuIHZhbHVlXG4gICAgICAucmVwbGFjZShcbiAgICAgICAgICAvW2Etel1bQS1aXS9nLFxuICAgICAgICAgIHYgPT4ge1xuICAgICAgICAgICAgcmV0dXJuIHYuY2hhckF0KDApICsgJy0nICsgdi5jaGFyQXQoMSk7XG4gICAgICAgICAgfSlcbiAgICAgIC50b0xvd2VyQ2FzZSgpO1xufVxuIl19
|